Ведущий Java Enterprise тренингов
14,5
рейтинг
23 января 2013 в 00:52

Разработка → Трудозатраты на реализацию «простого» модуля отправки Email в приложении с модульной архитектурой

На php отправка mail реализуется одной строчкой кода! А на java- нужно 3 недели??!
(из разговоров с разработчиками и менеджерами)


Статья не о том, как отправлять почту на java. Моя цель — показать сложности модульной разработки больших приложений (на примере разработки ERP River).

Итак, задача: реализовать сервис отправки по email (war).

Этапы разработки:



Начнем собственно с отправки

Если не Spring (для небольшого модуля он не нужен), подключаем apache commons-email
        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-email</artifactId>
            <version>1.3</version>
        </dependency>

и пишем
        public class MailSender {
            public static void sendMail {
                HtmlEmail email = ...
                ?
                email.send();

Позвольте, откуда брать настройки почтового сервера? Хардкодить их, думаю, не придет в голову даже младшему разработчику, поэтому:

Конфигурирование почтового сервера

Надеюсь, у нас уже выполнена общая работа по конфигурированию всей системы, и нам остается только реализовать ее для почтового модуля:
                    ...
                    <part key="email">
                        <entry key="hostName">smtp.gmail.com</entry>
                        <entry key="user">sendmail@mycompany.ru</entry>
                        <entry key="password">psw</entry>
                        <entry key="smtpPort">465</entry>
                        <entry key="useSSL">true</entry>
                        <entry key="debug">false</entry>
                        <entry key="charset">UTF-8</entry>
                        <entry key="useTLS">false</entry>
                    </part>                

        public class MailConfig {
            public static <T extends Email> T prepareEmail(T email) {
                 email.setHostName(hostName);
                 email.setSmtpPort(port);
                 email.setSSL(useSSL);
                 email.setTLS(useTLS);
                 email.setDebug(debug);
                 email.setAuthenticator(defaultAuthenticator);
                 email.setCharset(charset);
                 return email;
              }

Вызов сервиса

Ага, у нас сервис- как мы хотим его вызывать?
Бизнес хочет интеграцию по веб-сервисам, и нужно еще иметь отправку по простому HTTP GET (например, вызывать напрямую из браузера):
  • Отправка по HTTP GET:
            public class MailServlet extends CommonServlet {
                @Override
                protected void doProcess(HttpServletRequest request, HttpServletResponse response, Map<String, String> params) throws IOException, ServletException {
                     String from = ConfigUtil.getProperty("from", params);
                     ...
                     MailSender.sendMail(from, to, cc, ..);

  • Реализация вев-сервиса (JAX-WS) посложнее:

            @WebService
            @SOAPBinding(style = Style.RPC)
            public interface MailService {
                @WebMethod
                public void sendMail(
                    @WebParam(name = "from") String from,
                    @WebParam(name = "to") String to,
    
            @WebService(endpointInterface = "mycompany.MailService")
            public class MailServiceImpl implements MailService {
                @Override
                public void sendMail(String from, String to, String cc, String subject, String body, String attachmentUrls) throws StateException {
                    MailSender.sendMailAndRecordHistory(from, to, cc, subject, body, ..);
                }            

    и mailService.wsdl:
            <definitions ..
                    targetNamespace="http://mail.mycompany.com/" name="MailServiceImplService">
                <message name="sendMail">
                    <part name="from" type="xsd:string"/>
                    ...
    
                <portType name="MailService">
                    <operation name="sendMail" parameterOrder="from to cc subject body attachmentUrls">
                        <input wsam:Action="http://mail.mycompany.com/MailService/sendMailRequest" message="tns:sendMail"/>
                    ...
    
                <binding name="MailServiceImplPortBinding" type="tns:MailService">
                    <soap:binding transport="http://schemas.xmlsoap.org/soap/http" style="rpc"/>
                    <operation name="sendMail">
                        <soap:operation soapAction=""/>
                    ...
                <service name="MailServiceImplService">
                    <port name="MailServiceImplPort" binding="tns:MailServiceImplPortBinding">
                        <soap:address location="http://mycompany:8080/mail/mailService"/>
                    ...            

    Не забываем web.xml (Tomcat)
            <listener>
                <listener-class>com.sun.xml.ws.transport.http.servlet.WSServletContextListener</listener-class>
            </listener>
    
            <servlet>
                <servlet-name>mailService</servlet-name>
                <servlet-class>com.sun.xml.ws.transport.http.servlet.WSServlet</servlet-class>
                <load-on-startup>1</load-on-startup>
            </servlet>
            <servlet-mapping>
                <servlet-name>mailService</servlet-name>
                <url-pattern>/mailService</url-pattern>
            </servlet-mapping>
    
            <servlet>
                <servlet-name>mailServlet</servlet-name>
                <servlet-class>com.mycompany.mail.MailServlet</servlet-class>
                <load-on-startup>1</load-on-startup>
            </servlet>
            <servlet-mapping>
                ...            

Выделение mail-client

А как теперь соседнему модулю нашей системы быстро дернуть по веб-сервису наш сервис? Проще всего — выделить maven модуль mail-client, сделать от него зависимым наш mail сервис и разрешить любому модулю — нашему клиенту включать в себя (maven dependency) mail-client:
  • Делаем отдельный maven модуль mail-client и кладем в него mailService.wsdl и interface MailService
            <groupId>com.mycompany</groupId>
            <artifactId>mail-client</artifactId>
            <name>Mail Client</name>                

  • Кроме того, для полной радости нашего внутреннего клиента делаем MailWSClient:
    вызов соседнего модуля будет совсем простой:
    MailWSClient.sendMail(...
    

            public class MailWSClient {
                static String mailWsdl;
                private static final Service SERVICE;
    
                static {
                    URL url = MailWSClient.class.getClassLoader().getResource("mailService.wsdl");
                    SERVICE = Service.create(url, new QName("http://mail.mycompany.com/", "MailServiceImplService"));
                    // get mail endpoint from config
                    mailWsdl = Config.getUrlAsString("mail/mailService?wsdl");
                }
    
                public static void sendMail(String from, String to, ..){
                            getPort().sendMail(from, ..
    
                private static MailService getPort() {
                    MailService port = SERVICE.getPort(MailService.class);
                    Map<String, Object> requestContext = ((BindingProvider) port).getRequestContext();
                    requestContext.put(BindingProvider.ENDPOINT_ADDRESS_PROPERTY, mailWsdl);
                    return port;
                }                    

Прикручиваем шаблоны

Эге. В модуле документооборота у нас 52 вида документов. Хорошо б было нашим клиентам дать возможность самим определять шаблон письма. Тем более, что такой сервис (TemplateService) у нас уже реализован.
Сервис шаблонов простой: реализован на jsp, по get ему отправляются ключ и параметры, возвращается готовый текст.

  • Добавляем sendTemplateMail в MailService, MailWSClient, MailServiceImpl, MailSender, mailService.wsdl и MailServlet:
    
                sendTemplateMail(.., templateKey, params); 
                   
    
  • И реализуем его в MailSender (у нас уже есть удобная обертка MyHttpConnection, реализованная через HttpURLConnection.openConnection())
                static void sendTemplateMail(..., String key, String params) {
                    LOGGER.info("Send template mail from ...
                    String templateUrl = getUrlAsString("template?type=mail&format=html&key=" + key ...
                    MyHttpConnection conn = MyHttpConnection.connect(templateUrl, params);
                    if (conn.isOk()) {
                       String body = conn.getMsg();
                        sendMail(from, to, cc, MailUtil.getSubject(body), body);
                    } else {
                        throw LOGGER.getStateException(conn.toString(), ExceptionType.TEMPLATE);
                    ...                

    Попутно пришлось решить проблему с subject: сервис шаблонов возвращает только тело письма. Шаблон возвращается в формате html, MailUtil выделает из шаблона tiltle и использует его как subject:

            public class MailUtil {
                static Pattern MAIL_TITLE = Pattern.compile("<title>(.+)</title>", Pattern.MULTILINE);
    
                static String getSubject(String template) {
                    Matcher m = MAIL_TITLE.matcher(template);
                    return m.find() ? m.group(1) : null;
                }

Отправляем документ

Вообще-то у нас документы. А что, если вызывать нас сервис с id документа? Шаблоны для документов в TemplateService уже есть.
  • Добавляем sendDocMail в MailService, MailWSClient, MailServiceImpl, MailSender, mailService.wsdl и MailServlet.
    
    sendDocMail(String from, String to, String cc, String key, long docId);
                    
  • Опаньки, а у документов есть вложения, которые нужно аттачить к письму.
    К счастью commons-email это легко позволяет, и у нас есть общий maven модуль attach-common, у которого можно попросить список аттачей по docId:
            public class MailSender {
                static void sendDocMail(String from, String to, String cc, String key, long docId) throws StateException {
                    List<Attach> list = AttachUtil.getList(docId);
                    MailSender.sendTemplateMailAndRecordHistory(from, to, cc, key, "objectid=" + docId, MailUtil.formatAttach(list));
                }
    
            public class MailUtil {
    
                //  format attaches as
                //       ulr1[name1], ulr2[name2], ...
    
                static String formatAttach(List<Attach> list) {
                    return Util.collectionToDelimitedString(list, new Presentable<Attach>() {
                        @Override
                            public String toString(Attach attach) {
                            return AttachConfig.downloadUrl + attach.getUuid() + '[' + attach.getName() + ']';
                        }        

Отказоустойчивость

А если сервер временно недоступен? Нужно сохранять историю в базе и делать доталкиватель… Заодно решим проблему отправки письма пользователю по назначению на него задачи из BPM — ее можно будет реализовать через триггер в базе: вставлять в таблицу строчку TODO. Как side effect имеем историю отправки наших сообщений, можно потом сверху накрутить ui ну и просто SQL запросы к таблице поделать.
Хорошо, что у нас уже есть механизм сканирования — нужна просто еще одна ее реализация.


  • Делаем в базе таблицу mail_action
            CREATE TABLE hist.mail_action (
                id SERIAL,
                _from TEXT,
                _to TEXT NOT NULL,
                _cc TEXT,
                subject TEXT,
                body TEXT,
                attachmenturls TEXT,
                state TEXT NOT NULL,
                date TIMESTAMP(0) WITHOUT TIME ZONE,
                key reference.ui_key,
                params TEXT
            );
  • Добавляем в конфигурацию интервалы сканирования
            <entry key="scanTodoInterval">30</entry>
            <entry key="scanFailInterval">600</entry>    

    scanTodoInterval = ConfigUtil.getInt(SCAN_TODO_INTERVAL, mailProps, 60);  // default 60 sec
    scanFailInterval = ConfigUtil.getInt(SCAN_FAIL_INTERVAL, mailProps, 600); // default 10 min    

    Реализуем в MailSender запись истории отправки в базу вместе с состоянием (OK или Exception).
    Сканируем таблицу mail_action и на основе состояния state (TODO, EmailException) отсылаем письмо
            <listener>
                <listener-class>com.mycompany.common.web.SchedulerListener</listener-class>
            </listener>    

            public class MailWebScanner implements WebScheduler {
                private final MailScanner todoScanner = new MailScanner("TODO");
                private final MailScanner failScanner = new MailScanner("org.apache.commons.mail.EmailException");
    
                @Override
                public void activate(ServletContext servletContext) {
                    todoScanner.startScanning(MailConfig.scanTodoInterval);
                    failScanner.startScanning(MailConfig.scanFailInterval);
                }
    
                @Override
                public void deactivate() {
                    todoScanner.deactivate();
                    failScanner.deactivate();
                }
    
                @Override
                public void shutdown() {
                    AsyncExecutor.shutdown();
                }
            }
    
            public class MailScanner extends Scanner {
                private static final BeanListHandler<MailBean> HANDLER = new BeanListHandler<MailBean>(MailBean.class);
                private final String startWith;
    
                public MailScanner(String startWith) {
                    this.startWith = startWith;
                }
    
                void startScanning(int interval) {
                    activate(new Runnable() {
                        @Override
                        public void run() {
                            for (MailBean mail : getMailToSend()) {
                                MailSender.sendTemplateMailAndRecordHistory(
                            }
                        }
                    }, interval, false);
                }
                ...
    
                List<MailBean> getMailToSend() {
                    return SqlUtil.executeQuery("select * from hist.mail_action where state like '" + startWith + "%'", HANDLER);
                ...    

Для тех, кто не любит ждать: асинхронность

Так как наш сервис теперь устойчив к отказам, дадим возможность клиентам нашего веб-сервиса не ждать ответа. Вместо того, чтобы дублировать все методы серсвиса с постфиксом Async и аннотацией @OneWay добавим в вызовы MailWSClient флаг async и вызов AsyncExecutor (нашей обертки поверх ScheduledThreadPoolExecutor):
        public class MailWSClient {
            public static void sendMail(final String from, final String to, final String cc, final String subject, final String body, final String attachmentUrls, boolean async) throws StateException {
                send(new Runnable() {
                    @Override
                    public void run() {
                       getPort().sendMail(mask(from), mask(to), mask(cc), mask(subject), mask(body), mask(attachmentUrls));
                    }
                }, async);
            }

            public static void sendTemplateMail(final String from, final String to, final String cc, final String key, final String params, final String attachmentUrls, boolean async) throws StateException {
                ...


            public static void sendDocMail(final String from, final String to, final String cc, final String key, final long docId, boolean async) throws StateException {
                ...

            private static void send(Runnable task, boolean async) {
                if (async) {
                   AsyncExecutor.submit(task);
                } else {
                   task.run();
                }
             }

Чиним вложения картинок

Олично, все работает! Наконец, можно фиксить баги — картинки в письме не видны снаружи нашего интранета… Ведь они у нас в шаблонах заданы через <img src=«наши внутренние ресурсы», естественно, во всем остальном мире их не увидишь.

Делаем их встроенными:
        public class MailSender {
            static void sendMailAndRecordHistory(String from, String to, String cc, String key, String params, String attachmentUrls, long docId) throws StateException {
                ...
                String embedImgBody = MailUtil.embedImg(body, email);

        public class MailUtil {
            static final Pattern HTML_URL = Pattern.compile("<img src=(?:\"|')(.+)(?:\"|')", Pattern.CASE_INSENSITIVE | Pattern.MULTILINE);
            public static String embedImg(String body, final HtmlEmail email) throws EmailException {
                return StringUtil.resolveReplacement(body, HTML_URL, new Presentable<Matcher>() {
                    @Override
                    public String toString(Matcher matcher) {
                        String url = matcher.group(1);
                        cid = email.embed(url, UUID.nameUUIDFromBytes(url.getBytes()).toString());
                    }
                    return "<img src=\"cid:" + cid + "\"";
                 ...                

Отправляем встроенные (_img) большие картинки

Новая задумка бизнеса — по ошибке из браузера клиента отправлять на support mail скриншот экрана.
Решение на UI найдено — ход за нами. Для сервиса шаблонов пишем шаблон error_mail.jsp
            <%@page pageEncoding="UTF-8" %>
            <html>
            <head>
                <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
                <title>Error Report</title>
            </head>
            <body>
            <h2>Error Report from '${user}'</h2>
            <b>Message:</b>
                <pre>
                ${message}
                </pre>
            <b>Screenshot:</b><br>
            <img src="${screenshot}">
            </body>
            </html>
        

Параметры шаблона — exception message и base64_encoded_screenshot — отправляются в TemplateService из нашего сервиса. У нас проблемы: наша самописная обертка MyHttpConnection не может через GET отправлять base64_encoded_screenshot. Приходиться делать POST и еще раз делать URLEncoder.encode из за проблем с "+". Кроме того- в пришедшей почте inline картинка не видна :( Что ж, придется ее также делать вложением:
        public class MailUtil {
            static Pattern DATA_PROTOCOL = Pattern.compile("^data:(.+);(.+),");

            public static String embedImg(String body, final HtmlEmail email) throws EmailException {
                return StringUtil.resolveReplacement(body, HTML_URL, new Presentable<Matcher>() {
                @Override
                public String toString(Matcher matcher) {
                    String url = matcher.group(1);
                    String cid;
                    try {
                    Matcher m = DATA_PROTOCOL.matcher(url);
                    if (m.find()) {
                        final String cType = m.group(1);
                        final String encoding = m.group(2);
                        final String content = url.substring(m.toMatchResult().end());

                        cid = email.embed(new javax.activation.DataSource() {
                            @Override
                            public InputStream getInputStream() throws IOException {
                                try {
                                    return javax.mail.internet.MimeUtility.decode(new ByteArrayInputStream(IOUtil.getBytes(content)), encoding);
                                } catch (MessagingException e) {
                                    throw LOGGER.getIllegalStateException("Image encoding failed", e);
                                }
                            }
                            // empty realization for other javax.activation.DataSource methods
                            ...
                        }, UUID.randomUUID().toString());
                    } else {
                        cid = email.embed(url, UUID.nameUUIDFromBytes(url.getBytes()).toString());
                    }
                    return "<img src=\"cid:" + cid + "\"";
            ...
        

Финальная точка: безопасность

Однако, любой пользователь отправляет get запрос из браузера- и получает письмо с совершенно секретным документом. Нехорошо. Необходимо прикрутить проверку доступа у пользователя к документу с переданным docId и вообще проверить: если запрос пришел по по get, залогинен ли пользователь в нашу систему.
Из-за того, что страница с логином уже была сделана и вокруг нее много что вертелось, а точка входа в систему у нас одна, я сделал проверку через REST и куки уровня домена с доверием к серверным запросам между самими модулями, но это уже — отдельная статья.

Итоги простой задачи отправки почты:

В результате получилось 2 maven модуля с классами (не считая инфраструктуры типа конфигурации, вложений, шаблонов, общей части и JUnit тестов)
  • mail-client
    • MailService: интерфейс (sendMail, sendTemplateMail, sendDocMail)
    • MailWSClient: обертка к клиенту, выставляющая endPoint из конфигурации
    • mailService.wsdl

  • mail-service
    • MailSender: собственно отправка
    • MailServiceImpl: имплементация веб-сервиса, делегирование в MailSender
    • MailServlet: сервлет для обработки HTTP GET
    • MailBean: бин для чтения строки из базы через commons-dbutils
    • MailConfig: конфигурация
    • MailScanner: сканирование таблицы по состоянию отправки
    • MailWebScanner: реализация листенера для нашего сервиса, запускающего 2 сканнера MailScanner
    • MailUtil: утильные методы
    • EmailExceptionHandler: обработка exceptions, не доталкивается AddressException
    • sun-jaxws.xml, web.xml


Терпеливый читатель, дошедший до конца статьи может сам сравнить количество трудозатрат на «задачу отправку почты» и полученную реализацию. Спасибо за внимание.



Ссылки:

Григорий Кислин @gkislin
карма
18,0
рейтинг 14,5
Ведущий Java Enterprise тренингов
Реклама помогает поддерживать и развивать наши сервисы

Подробнее
Реклама

Самое читаемое Разработка

Комментарии (74)

  • +46
    Энтерпрайзненько, однако
  • +6
    Да на PHP тоже нужно 3~ недели, если хотите чтоб было по качеству так-же как и у вас. Да и еще и самому протестировать.
    Одна строка, может быть и в java, если вызвать консоль с мейл демоном и передать демону нужные параметры, качество оставит желать лучшего.
    • 0
      Ну… Тут же используется модуль. Если для php тоже взять модуль, вроде SwiftMailer, то три недели не нужны.
  • +2
    спасибо за html2canvas)
  • +20
    Бывает. Накипело. Пора создавать клубы анонимных программистов.
  • +6
    С каждым годом вычислительные мощности компьютеров растут, но для компенсации прироста производительности хитрые разработчики придумали фреймворки.
    • +13
      Вот в этой задаче пригодился бы ассемблер!
  • –2
    Не плохо. Но на Django + Celery + django-celery-email + немного своей логики было бы проще. Да, не было бы внутренних SOAP сервисов, мы просто не используем их. Аутентикация на security декораторах. Шаблониатор и прочее как dependency injection подбираются в рантайме. Да, асинхронно. Да, если задача отвалилась, перезапуститься с дефолтным retry=5, иначе ругаться к админу мылом. Да, с аттачами. Да, в аттачах что угодно. Да быстро. Да scalable. Да, буквально 1-2 дня.

    Не скажу, что подход на Java как то хуже. Просто он и не лучше. Всего то надо попробовать что-то другое, что бы это увидеть.
    • 0
      А через API Outlook еще проще. И шо?)
      • +1
        При чем тут это? Сложность не в самой отправке, она одной строчкой. Сложность в отказоустойчивости, видимости (каждая задача для Celery видна, можно перезапустить руками или автоматом), удобной интеграции с другими компонентами системы (шаблонизатор, аттачи/документы и т.п.) и удобном использовании компонента, в scalabilty. Так вот это все можно сделать и делают проще. Вместо 3 недель 1-3 дня.
        • –1
          По моему мнению, сравнивать пайтон с джавой некорректно — разный класс. Пайтон можно сравнивать с пхп.
          • +2
            Обоснуйте? У решения есть некоторые качества, такие как корректное поведение под нагрузкой, scalability, корректная обработка всех edge cases, интеграция в другие компоненты (компонент — не одинокий джекичан). Разве важно на каком языке это все было написано?

            Другое дело, что архитекура приложения иная. Нет множества SOAP сервисов, что работают вместе и каждый по отдельности это bottle neck, который еще и отдельно нужно скейлить если превышаем нагрузку. Если сервис был написан через пятую точку (более чем реально), то скейлить на несколько машин не получиться, требуется общая VM память/объекты, общие локи и прочие прелести безумного кодинга. В итоге один сервис, джекичан, тянет всех вниз.

            На питоне все компоненты работают в одном application server. А вот этих серверов может быть сколько угодно. Байткод не занимает особо памяти, поэтому потребление памяти на один app server небольшое. Впереди серверов стоит один/пара front-end серверов, работающих как буфферы (nginx). Все асинхронные задачи выносятся в Celery/подобное, локов просто нет, все сервера stateless (состояние выносится в базу/Redis/RabbitMQ/по задаче). Celery скейлиться на несколько отдельных машин, если задач много. При желании динамически. Рассылка мыла обычно выносится на отдельный MTA, само приложение об этом даже не знает.

            И все это не что-то типичное для питона. То же самое можно построить на Perl, PHP или Java. Вопрос не языке, а в архитектуре всего приложения. И в необходимом времени. Оно все может занять 3 недели, если писать морду на админку, где видны все задачи в микродеталях, маниакально детальная статистика с десятком графиков. Но я этого из статьи не увидел. Да и ортогонально это все как-то основной задаче.
            • +3
              Топ3 решений на пайтоне для энтерпрайза? Их объем рынка?
            • 0
              Спасибо, интересная вещь, нужно будет почитать.
              На java тоже есть асинхронный запуск задач (akka, hazelcast), главное- не стрелять из пушек по воробъям.
              Soap — он как раз для разнесения серверов и языконезаисимости.
              Аутентикация на декоратарах не сильно проще ее же в фильтре будет…
              Питон- хороший язык, но я люблю статическое типизирование. Поэтому счас смотрю в сторону Scala…
    • 0
      Вы предлагаете автору переписать проект на Django? :-)
      • –2
        Зачем, сравенние — всегда вещь полезная. Я писал на Django совсем немного (и давно), поэтому мое мнение о нем очень субъективное- по поему он мало-объектоно-ориентирован. Фунциональность добавляется через hook-и. Я предпочитаю template method. То же могу сказать о работе в нем с базой.
  • +10
    Эта картинка должна быть здесь: problem-factory.jpg.to

    Простите.
    • –1
      :))
  • +8
    На php отправка mail реализуется одной строчкой кода! А на java- нужно 3 недели??!


    Java, первая ссылка в гугле (отправка gmail):

    public static void Send(final String username, final String password, String recipientEmail, String ccEmail, String title, String message) throws AddressException, MessagingException {
            Security.addProvider(new com.sun.net.ssl.internal.ssl.Provider());
            final String SSL_FACTORY = "javax.net.ssl.SSLSocketFactory";
    
            // Get a Properties object
            Properties props = System.getProperties();
            props.setProperty("mail.smtps.host", "smtp.gmail.com");
            props.setProperty("mail.smtp.socketFactory.class", SSL_FACTORY);
            props.setProperty("mail.smtp.socketFactory.fallback", "false");
            props.setProperty("mail.smtp.port", "465");
            props.setProperty("mail.smtp.socketFactory.port", "465");
            props.setProperty("mail.smtps.auth", "true");
    
            props.put("mail.smtps.quitwait", "false");
    
            Session session = Session.getInstance(props, null);
    
            // -- Create a new message --
            final MimeMessage msg = new MimeMessage(session);
    
            // -- Set the FROM and TO fields --
            msg.setFrom(new InternetAddress(username + "@gmail.com"));
            msg.setRecipients(Message.RecipientType.TO, InternetAddress.parse(recipientEmail, false));
    
            if (ccEmail.length() > 0) {
                msg.setRecipients(Message.RecipientType.CC, InternetAddress.parse(ccEmail, false));
            }
    
            msg.setSubject(title);
            msg.setText(message, "utf-8");
            msg.setSentDate(new Date());
    
            SMTPTransport t = (SMTPTransport)session.getTransport("smtps");
    
            t.connect("smtp.gmail.com", username, password);
            t.sendMessage(msg, msg.getAllRecipients());      
            t.close();
        }
    


    PHP, первая ссылка в Google отправка через gmail:

    <?php
    
           require_once "Mail.php";
    
            $from = "<from.gmail.com>";
            $to = "<to.yahoo.com>";
            $subject = "Hi!";
            $body = "Hi,\n\nHow are you?";
    
            $host = "ssl://smtp.gmail.com";
            $port = "465";
            $username = "myaccount@gmail.com";  //<> give errors
            $password = "password";
    
            $headers = array ('From' => $from,
              'To' => $to,
              'Subject' => $subject);
            $smtp = Mail::factory('smtp',
              array ('host' => $host,
                'port' => $port,
                'auth' => true,
                'username' => $username,
                'password' => $password));
    
            $mail = $smtp->send($to, $headers, $body);
    
            if (PEAR::isError($mail)) {
              echo("<p>" . $mail->getMessage() . "</p>");
             } else {
              echo("<p>Message successfully sent!</p>");
             }
    
        ?>  <!-- end of php tag-->
    


    Вполне сопоставимо по сложности. Преимущество этого Java-кода — он более универсален, можно использовать и для консольных, и для десктопных и для Web. PHP более хардкорный.

    Хотя ваша идея абсолютно ясна: задачи с одним и тем же названием могут отличаться по сложности в тысячи и десятки тысяч раз. Мой любимый пример — логгер. Самый простой способ логировать в C# — одна строчка: File.Write (ну 2 строчки, если нужен потокобезопасный). Однако почему то возникла необходимость в создании таких продуктов как log4net и NLog — а там работы на несколько человеко-месяцев.
    • +3
      PHP скрипт читерский — заюзали pear библиотеку
    • НЛО прилетело и опубликовало эту надпись здесь
  • НЛО прилетело и опубликовало эту надпись здесь
    • –2
      1. Локальный сервер тоже может лежать.
      2. SMTP-сервер может просто не справляться с нагрузкой.
      3. Не все приложения работают на одном компьютере.
      • +8
        1. Даже если MTA лежит, исходящая почта будет падать в очередь и отправится как только MTA встанет.
        2. Как следует из п.1, при любой нагрузке почта теряться не будет, отправится рано или поздно. Ну и предпосылка о том, что удаленный SMTP-сервер уж точно справится с нагрузкой, несколько сомнительна.
        3. MTA можно поставить на каждый сервер и указать всем, например, единый смартхост, через который они будут пулять исходящую почту. Абсолютно нормальное решение. Уж точно лучше попытки сделать свой отказоусточивый велосипед :)
      • НЛО прилетело и опубликовало эту надпись здесь
    • –1
      Интересное решение.
      Но — из триггера в базе уже не отправишь. Так получилось универсально.
      К тому же на доталкивание можно логику повесить.
      • НЛО прилетело и опубликовало эту надпись здесь
  • +2
    Ну Вы смешали задачи совершенно не относящиеся к отправке почты: сервлеты, веб-сервисы, шаблоны (кстати я использую отдельный шаблон для темы и предпочитаю freemarker). Но, в целом, Вы правы, это не простая задача. Например для Grails я разработал плагин Asynchronous Mail, который позволяет отправлять почту в одну строчку :). Но он тянет за собой 3 плагина, которые тянут 2 фрэймворка.
  • –1
    «б.., это… здец»
  • +1
    для ведения промышленной разработки на РНР, также используеся модульная архитектура,
    а тем более рассылка — дело довольно-таки не тривиальное… И чтоб сделать толковый модуль и внедрить его в существующую систему, даже используя готовые классы PHPMailrr или Pear потребуется ни одна пара-тройка дней.
  • +1
    Да уж, чего только не придумают, чтобы не набрать

    yum install postfix
    

    ;-)
    • 0
      И что это дает, кроме инсталляции постфикса?
      • +9
        Вся статья сведется к вызову mail (sendmail) и скармливанию заполненного шаблона. Ну и обертке jax-ws к этому (или сервлету? что там автору надо-то).

        И недоступность отработает, и доставку, и очереди. И scalability будет такое, что самопальным наколеночным поделиям и не снилось. А самое смешное — не понадобится ни конфигурация (зачем? на отсылку-то стандартный смартхост из коробки!), ни эм-бинов с БД, ни прочих ентерпрайзиков.

        Заодно и узнаете, что mail client это нечто другое ;-)

        Посреди многослойной, переплетенной из еаров, сервисов, коннекторов и пулов структуры сервера приложений, органично врастая в сеть модульной архитектуры, выставив наружу красивые интерфейсы, посылая и получая сообщения, отвечая на эрэмай запросы, словно кипящий поток создавая и уничтожая сотни ентити бинов в десятках распределенных транзакций, мудрый сессионный бин срал в лог эксепшенами.


        • 0
          1. Нужна отсылка по вставке записи в базу (BPM создает запись, когда пользователю назначается задание). Как вы ее предлагаете решить через postfix?
          2. Окружений у нас — 6 штук. Бывает гораздо больше. Услилия на «самопальное наколеночное поделие»- ксати достаточно простое- очень даже сравнимы на инсталирование MTA (переконфигурирование, поддержание на всех окружениях)
          2. Конфигурация в многослойном таком приложенни — есть. Такой факт. Я сторонник держать все ее в одном месте, а не распылять по конфигам. Только если без этого не обойтись. Например- по умолчанию отправка от чъего имени, или суппорт майл — настройки почты, но никак MTA не отдать.
          PS: а последняя фраза- красивая, не спорю.
          • +11
            (смотрит) ну давайте мы внезапно еще что-нибудь узнаем про вашу задачу.

            Про окружение посмеялся. Да-да, а перенастраивать все остальное как бы не надо. Само заработает.

            * * *

            У меня для вас есть рассказ — в далеком 1995м году я был такой же молодой и горячий. И тоже рисовал наколеночную отсылку почты, ровно в таком же ентерпрайзненьком стиле. Все было замечательно, пока не поставили этого кадавра клиентам. И тут! Тут началось.

            Однажды, в пятницу вечером, то ли гроза выбила сетевушку, то ли уборщица зацепила шваброй провод — но оно таки случилось. И робот по отсылке почты тут же стал слать алерты — ну как же, отвалилось, примите меры. Отсылать было некуда, и все это ровно по вашему рецепту складывалось в БД.

            Однако, за субботу с воскресеньем было налеплено столько писем, что на диске закончилось место. БД с логами за него подрались. Поинтересуйтесь, что в этом случае делает MySQL — вам понравится. А поскольку кроме почты БД пользовались и другие — оно все тоже встало колом.

            И вот в понедельник с утра — пришел дежурный инженегр. Увидев, что все встало, он подцепил сетевой шнурок. Однако это не помогло. Обнаружив что на диске закончилось место, он удалил что-то ненужное.

            Это не конец истории — дальше было вот что. Могучий досылатель, радостно заметив что почтовый сервер появился, тут же «дослал». SMTP там стоял какой-то древний демон на новеле, и от такого подарка он тут же сдох. Ну правильно — теперь на почтовом сервере место закончилось.

            Новелевское чудо тоже было в ентерпрайзненьком стиле — и пришлось выгребать помои руками. Сначала из очереди, а потом из мейлбоксов — часть спама таки доехала.

            Что же было дальше? А вот что, досылатель снова обнаружил что почтовый сервер работает, и снова дослал новую порцию (да, я был креативным, и не пытался все письма залить сразу. Я разбил на порции. Предусмотрительный был, ага. А то бы сдох кадавр, и проблему решили бы быстрее).

            Бам!

            В понедельник ближе к ночи была найдена причина этих «бам». Разумеется, это был наш кадавр — но попробуй это определить, когда у тебя серверная далеко не из двух компов состоит, а начальство размахивает руками над ухом.

            И вот под утро вторника кадавру почистили очередь руками в базе, не разбираясь — а было ли там хоть что-то полезное, база данных отсвопалась после освобождения места на диске, и все это чудо наконец-то заработало в штатном режиме.

            А через месяц вечером к большому боссу клиента пришел старинный приятель. И потребовал чаю. И секретарша, не найдя розетки поближе, выдернула раутер и включила чайник. О ее умственных способностях она узнала только утром. Чайник выдернули, вернули ценный аппарат на место… но кадавр-то не обесточен!

            БАМ!!!

            Нет, второй раз починили быстрее, к обеду. Кто-то вовремя догадался про могучий досылатель, ведь история была еще свежа в памяти. И вот вместо этого кошерного ентерпрайзика мои старшие товарищи, вспоминая через раз про мои умственные способности, в субботу вечером лепили на коленке простую отсылалку, ориентированную на использование стандартных средств. А я, как самый хитрый, сидел рядом и краснел учился.

            В новой инкарнации, отсылатель почты отработал десять лет без единого сбоя. Менялась инфраструктура, сервера и версии софта, но древний монстр, написанный на java 1.0.2, все так же тихо работал на древнючем ультраспарке под не менее древним SunOS.

            К слову, 10 лет спустя я был свидетелем демонтажа этого спарка. Единственный раз в жизни я видел слой пыли в системнике, который слежался до такой степени, что из него сразу можно было делать валенки. Первая мысль электронщика была «а зачем в корпус кусок пальто положили?»

            Вы говорите — поцфикс не решает? ;-)
            • 0
              Интересная история, спасибо. Тоже посмеялся. Ну и есть чему поучиться на чужих ошибках.
              ps: поставлю на аварийную отсылку ограничитель. В уме это конечно было, даже озвучил- но отклика вверху не нашло, поэтому оставил. А тут вижу- не надо лениться.
              • НЛО прилетело и опубликовало эту надпись здесь
                • 0
                  Не читал, спасибо. Делал я как то SMS сервис (тоже в SOA). Все гораздо сложнее:)
                  Если это конечно действительно ентерпрайз.
                  1. Нужно отделить интерфейс от реализации- в разных странах разные провайдепы, да и внутри себя иногда их меняют.
                  2. Было требование на ограничение по времени и количеству на одного клиента и на систему в целом- (в день- не больше 200, в минуту- не больше 5 например). Причем задаваемое в конфигурации. Не совсем простая задача, когда начинаешь реализовывать.
                  3. Нужны были счетчики по событиям и статистика. Не меньше этого сервиса получилось:)
            • +1
              Напомнило: В одной из больниц Йоханнесбурга (ЮАР) стали повторяться необъяснимые случаи смерти пациентов после операций — серьезных, но прошедших, казалось бы, вполне успешно. Причем все такие случаи происходили по пятницам.

              Расследование показало, что в пятницу в отделение хирургии приходила уборщица. Убирая в реанимационной палате, она ничтоже сумняшеся выдергивала вилку установки «искусственное сердце» из розетки в стене и включала свой пылесос… (http://www.nkj.ru/archive/articles/4083/?sphrase_id=83855)
              • +1
                Интересно, сколько человек вам поверило?
            • 0
              Это достойно отдельного топика -) На хабре нет раздела «байки»? :)
  • 0
    Если известно, что приложение будет исполняться в типичном UNIX-окружении, то, несмотря на обратное утверждение автора статьи, настройки SMTP-сервера можно и захардкодить: 127.0.0.1:25 без авторизации, т.к. в рассматриваемом окружении это всегда правильно. Это уже дело локального почтового сервера переслать письмо через правильный почтовый релей с авторизацией под правильной учетной записью.
    • 0
      Хардкодить в принципе не правильно. Да, можно зарелеить автоматом, но перед отправкой локальный серврер может сбрасывать мыло на диск, чтобы в максимально короткое время принятъ мыло полностью и освободить клиента. А если обзваниваем большую базу клиентов, да с аттачами, это может быть весьма весомо. При криворукости админа система начинает жевать своп, все замедляется тысячи раз, отправка нового мыла заморожена. Если мыло-сервер еще и на одной машине с чем то важным (база, app server, все зависит от фантазии админа), то и эти сервисы замораживаются. Всякие сценарии возможны.
      • 0
        Хотя, все таки нереально это как то все. Извиняюсь за коммент, не могу удалить.
    • 0
      Правильнее все-таки дать возможность настроить smtp, но предусмотреть умолчания — localhost:25 и без пароля.
  • 0
    Если известно, что приложение будет исполняться в типичном UNIX-окружении, то, несмотря на обратное утверждение автора статьи, настройки SMTP-сервера можно и захардкодить: 127.0.0.1:25 без авторизации, т.к. в рассматриваемом окружении это всегда правильно. Это уже дело локального почтового сервера переслать письмо через правильный почтовый релей с авторизацией под правильной учетной записью.

    Или тупо вызвать sendmail, который работает (складывает письмо в очередь) даже если собственно smtp-демон в данный момент перезапускается. Лишь бы хватало места и inodes.
  • +1
    Тут поступил приватный коммент от Elmot- некаширно использовать GET для действий — можно получить спам: forums.asp.net/t/1822324.aspx/1
    В оправдание скажу только что это все таки интранет приложение и чз GET из браузера сделать ссылку проще. Но в целом- согласен. Можно подумать на тему- оставить только post.

    • +4
      Да уж, столько понаписали, и такой косяк с GET-ом. В этом вся Java — куча настраиваемых и конфигурируемых (недельки за три) компонентов, за которыми не видно базовых вещей.
      GET здесь совершенно, ну совершенно не в кассу.
      Объясню на пальцах: вы, вообще говоря, не контроллируете, как браузер или другой http-клиент будет использовать GET-методы, благо стандарт HTTP, особенно если его «вольно трактовать» и «оптимизировать», это позволяет. Клиент может, если это старый ИЕ или по дороге есть кривой прокси, вообще не вызывать GET — а фигли, закешировано же.
      Ну и главное, конечно, безопасность. Отправляем вашему залогиненному юзеру например html емейл с картинкой <img src='ваш get url'> и всё — почта отправится «сама».
      Для подобных действий — только POST. Для хардкорных html-кодеров есть формы, которые ничуть не сложнее просто ссылок; для js вызов get и post вообще ничем не отличаются, кроме названия метода.
      • НЛО прилетело и опубликовало эту надпись здесь
        • 0
          Действительно- еще не пользовал. Но в данном случае- он неуместен:)
          • НЛО прилетело и опубликовало эту надпись здесь
        • 0
          Всё же у JAX-RS и JAX-WS цели несколько разные. И когда нужна тонна энтерпрайзных стандартов WS-*, то REST нервно курит в сторонке, к сожалению.
          • НЛО прилетело и опубликовало эту надпись здесь
            • 0
              Ну да, хоть и не соответствует идеологии REST, т. к. отнюдь не является идемпотентным действием.
              • НЛО прилетело и опубликовало эту надпись здесь
                • 0
                  Технически — да.

                  JAX-RS — это Java API for RESTful Web Services, так что идеологически JAX-RS предназначен для RESTful сервисов, что и задает рамки.

                  Поэтому REST-подход вполне ожидаем (и вызывает наименьшее удивление) при использовании JAX-RS. Я говорю исключительно про это.

                  А то, что в каком-то сервисе GET-запрос будет являться не идемпотентным, может дать дыру в безопасности — другая сторона вопроса.
                  • НЛО прилетело и опубликовало эту надпись здесь
                    • 0
                      Можно посмотреть и с этой стороны. Но в случае Java JAX-RS наиболее подходит для RESTful сервисов.

                      Как вспомню restlet'ы, аж передёргивает.
              • НЛО прилетело и опубликовало эту надпись здесь
      • –2
        Аутентификацию прикрутить и всё.
        Да и откуда инфа, что GET этот будет из браузера вызываться а не из какого-то клиента в качестве RESTful сервиса.
        И заменить GET на POST в том коде — дело двух секунд.
        • 0
          Аутентификация не поможет, если юзер уже залогинен на атакуемом сервисе и все нужные куки у него уже стоят. Пример — в одной вкладке браузера открыт атакуемый сервис, в другой — веб-почта с веселыми картинками. CSRF токены тоже могут не справиться, хотя спасут хотя бы от повторяемых множественных атак (скажем, от десятка «картинок» в одном письме).
          Смысл в том, что использовать в данном случае GET вместо POST — это всё равно, что использовать SELECT… PROCEDURE вместо INSERT для создания записей в базе данных или забивать шурупы молотком. Теоретически возможно, но абсолютно безграмотно.
        • 0
          Аутентификация есть. Но от ссылки в мыле залогиненного пользователя- не спасет. Post тоже есть в CommonServlet (см.
          public class MailServlet extends CommonServlet). Запретить GET- ерунда. Только клиена UI еще придется с get на post перевести, ну и JUnit-ы поправить.
      • 0
        И на старуху бывает проруха…
  • 0
    >А если сервер временно недоступен? Нужно сохранять историю в базе и делать доталкиватель…

    Вообще-то с этого надо было начать — при нажатии кнопки «Отправить», создается запись в базе.
    Ну а потом по базе бегает сервис, смотрит, что нужно отправить ну и отправляет.
    • 0
      Записывается в блоке final. С результатами отправки.
  • 0
    А тебе, пожалуйста, конкретное требование заказчика в студию.
    • 0
      *теперь, простите
      • 0
        Заказчик фичи- бизнес (аналитики).
        Отправлять почту (желательно по веб-сервису). С доталкивателем.
        + позже — по ошибке из браузера клиента отправлять на support mail скриншот экрана.

        Детали реализации, интеграция с имеющимися модулями и с окружением — тут уж я сам… Например, самому сходить за шаблоном: удобнее пользоваться будет, асинхронность, конфигурирование и пр.
  • 0
    Я теперь буду показывать эту статью людям, говорящим «ничего сложного» :)
  • 0
    А Вы все-таки смухлевали! Вместо apache commons надо было использовать труЪ JavaMail API ;))
    • 0
      Это вы к изобретению собственных велосипедов? Я сторонник пользоваться готовым.
      • 0
        Да нет, вроде как стандарт: www.oracle.com/technetwork/java/javamail/index.html
        А commons — это как раз велосипед. Просто на порядок сложнее.
        • 0
          Вы заглянули по первой ссылке… commons-email — удобные обертки к javamail.

Только зарегистрированные пользователи могут оставлять комментарии. Войдите, пожалуйста.