ErmesMail is a set of Java classes for sending e-mail messages asynchronously, via SMTP servers.
- It can be embedded in your Java project as a tiny wrapper for the Apache Commons Email library.
- it can be used as a handy command line utility, to send emails programmatically from the shell.
ErmesMail is developed in Java 17 and built with Maven 3.8.
JavaDocs are available here.
- Build the application with maven:
mvn package
. - Run the Java executable by passing the following parameters:
$ java -jar target/ermes-mail.jar --help
Usage: java -jar ermes-mail.jar [-v] [--help] [--sslon] [-P[=<password>]]
-b=<message> -f=<fromAddress> [-h=<smtpHost>]
[-n=<senderName>] [-p=<smtpPort>] -s=<subject>
[--sslport=<sslPort>] [-u=<user>]
[--bcc=<bccList>[,<bccList>...]...]...
[--cc=<ccList>[,<ccList>...]...]...
--to=<toList>[,<toList>...]... [--to=<toList>[,
<toList>...]...]...
Sends an HTML email to the given recipient(s).
-h, --host=<smtpHost> SMTP host.
-p, --port=<smtpPort> SMTP port.
-u, --user=<user> SMTP user name.
-P, --password[=<password>]
SMTP user password.
--sslon Use SSL.
--sslport=<sslPort> SSL port (default is 465).
-f, --from=<fromAddress> FROM field.
-n, --sender=<senderName> Sender full name (optional).
-s, --subject=<subject> Subject.
-b, --body=<message> Message body (can be HTML).
--to=<toList>[,<toList>...]...
List of mandatory TO recipients.
--cc=<ccList>[,<ccList>...]...
List of optional CC recipients.
--bcc=<bccList>[,<bccList>...]...
List of optional BCC recipients.
--help display this help message.
-v, --version print version information and exit.
Copyright(c) 2022 SoftInstigate srl (https://www.softinstigate.com)
To test the sending of e-mails via comand line, we suggest running a local SMTP mock server like MailHog. Please look here for MailHogs's installation instructions.
After executing MailHog (usually with the MailHog
command) you can send your first HTML email message to localhost
with ErmesEmail:
$ java -jar target/ermes-mail.jar -h localhost -p 1025 \
-f sender@email.com -s "test" -b "This is a <strong>HTML</strong> test email." \
--to receiver@email.com
mag 24, 2022 4:46:16 PM com.softinstigate.ermes.mail.EmailService <init>
INFORMAZIONI: MailService initialized with SMTPConfig{hostname='localhost', port=1025, username='', ssl=false, sslPort=465}
mag 24, 2022 4:46:16 PM com.softinstigate.ermes.mail.EmailService send
INFORMAZIONI: Sending emails asynchronously...
mag 24, 2022 4:46:16 PM com.softinstigate.ermes.mail.SendEmailTask call
INFORMAZIONI: Processing MailModel{from='sender@email.com', senderFullName='null', subject='test', message='This is a <strong>HTML</strong> test email.', to=[Recipient{email='receiver@email.com', name='null'}], cc=[], bcc=[], attachments=[]}
mag 24, 2022 4:46:16 PM com.softinstigate.ermes.mail.SendEmailTask call
INFORMAZIONI: Email successfully sent!
TO: [Recipient{email='receiver@email.com', name='null'}]
CC: []
BCC: []
mag 24, 2022 4:46:16 PM com.softinstigate.ermes.mail.EmailService shutdown
INFORMAZIONI: ExecutorService terminated normally after shutdown request.```
You can read the e-mail message on the MailHog UI.
Note: To send messages via Google SMTP, it is necessary to configure your Gmail account by enabling IMAP. More information
To use ErmesMail in your Maven build, first add the JitPack repository in your pom.xml
<repositories>
<repository>
<id>jitpack.io</id>
<url>https://jitpack.io</url>
</repository>
</repositories>
Then add the following dependency:
<dependency>
<groupId>com.softinstigate</groupId>
<artifactId>ermes-mail</artifactId>
<version>1.1.0</version>
<classifier>shaded</classifier>
</dependency>
Warning: As ErmesMail depends on
org.apache.commons.commons-email
v1.5, we suggest to include the below runtime dependencies (javax.mail-api
andjavax.mail
) to prevent classpath conflicts. The wrong version of these dependancies, included by other libraries, might provoke the following runtime exception when sending emails:java.lang.NoSuchMethodError: 'void com.sun.mail.util.LineOutputStream.<init>(java.io.OutputStream, boolean)
<dependency>
<groupId>javax.mail</groupId>
<artifactId>javax.mail-api</artifactId>
<version>1.5.6</version>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>com.sun.mail</groupId>
<artifactId>javax.mail</artifactId>
<version>1.5.6</version>
<scope>runtime</scope>
</dependency>
You can run mvn dependency:tree
in your project to check if other artifacts are including these.
There are two methods for sending emails: EmailService.send
is asynchronous and returns a Future list of error strings. the EmailService.sendSynch
is synchronous and returns a list of error strings. If the list is empty, it means no errors. However, the SendEmailTask logs exceptions anyway.
You may wanto to use the asynchronous invocation only in case you have to send tons of email in parallel and don't want to block the rest of the program, otherwise the synchronous method works just fine.
Internally the EmailService uses a java.util.concurrent.ExecutorService to send emails in parallel.
A good understanding of Java Futures would help you implementing the best waiting strategy.
Below a java fragment, for example:
SMTPConfig smtpConfig = new SMTPConfig("localhost", 1025, "user", "password", false);
EmailModel emailModel = new EmailModel(
"dick.silly@domain.com", "Dick Silly",
"Test email - " + System.currentTimeMillis(),
"This is a <strong>HTML</strong> message.");
emailModel.addTo("john.doe@email.com", "John Doe");
emailModel.addTo("serena.wiliams@email.com", "Serena Wiliams");
emailModel.addCc("tom.clancy@email.com", "Tom Clancy");
emailModel.addBcc("ann.smith@email.com", "Ann Smith");
EmailService emailService = new EmailService(smtpConfig, 3); // 3 threads pool
Future<List<String>> errors = emailService.send(emailModel); // send is asynch
emailService.shutdown();
List<String> listOfErrors = errors.get(); // WARNING: Future.get() is blocking
if (!listOfErrors.isEmpty()) {
System.err.println("Errors sending emails: " + listOfErrors.toString());
}