Warning! This documentation is a work in progress. Expect things to be out of date and not actually work according to instructions.

Sending Email

First you need to define an SMTP server in your stallion.toml settings. You can get an SMTP email sending service for free from services such as Postmark, Sendgrid, or Mailgun.

[email]
host = "smtp.postmarkapp.com"
username = "yourusername"
password = "yourpassword"
defaultFromAddress = "someone@stallion.io"
adminEmails = ["admin@stallion.io"]
port = 587
canSpamText = "Sent by Stallion Software LLC 123 AnyStreet, Anytown USA"

(Remember, that if you do not want to put your password in plain-text, you can use encrypted secrets management).

Here is how to send a basic email using io.stallion.email.EmailSender, which itself just wraps javax.mail.

var EmailSender = Packages.io.stallion.email.EmailSender; var emailer = EmailSender.newSender(); emailer .setFrom(transformMaybe(getFromAddress())) .setHtml(html) .setReplyTo(transformMaybe(getReplyTo())) .setSubject(transformMaybe(getSubject())) .setTo(user.getEmail()); EmailSender emailer = EmailSender.newSender(); emailer .setFrom("me@service.com") .setHtml(html) .setReplyTo("service@service.com") .setSubject("You are the luck winner!") .setTo("customer@somewhere.com") .send();

The option above is very basic. There is a more complete option that gives you a bunch of nice helpers, and can be used for sending an email to any object that implements the Contactable interface (such as a User or a Contact).

Use it by creating a subclass of the io.stallion.email.ContactableEmailer

Features include:

  • easy support for sending emails based on Jinja templates
  • respect default user email opt-out
  • safe-guards for preventing accidentally sending duplicate emails

Here is an example of the built-in password reset emailer:

var JsExtendableContactableEmailer = Packages.io.stallion.email.JsExtendableContactableEmailer; var EventEmailer = Java.extend(JsExtendableContactableEmailer, { // If true, ignore default user email opt-out isTransactional: function(self) { return false; }, // Get the jinja template for the email getTemplate: function(self) { return 'emails/comment.jinja'; }, // Who is sending the email getFromAddress: function(self) { return '{ commenter.displayName } via UpFor '; }, // The email subject getSubjectJs: function(self) { return 'RE: ' + self.get('event').emailSubject; } }); // Send one email to a particular user: (new EventEmailer(user, {event: event, commenter: commenter})).sendEmail(); public class ResetEmailEmailer extends ContactableEmailer { public ResetEmailEmailer(T user, String resetToken, String returnUrl) { super(user); // Add variables to the template context put("resetToken", resetToken); put("returnUrl", url); } /** * As a transactional email, will be sent even if the user has opted out of * of marketing emails */ @Override public boolean isTransactional() { return true; } /** * Get the template for this email */ @Override public String getTemplate() { return "stallion:email/reset-password.jinja"; } /** * The subject is a simple template string that can interpolate context variables */ @Override public String getSubject() { return "Reset your password for {user.username} on " + Settings.instance().getSiteName(); } /** * Unique keys get stored in memory, and ensure we never send the same email twice */ public String getUniqueKey() { return truncate(GeneralUtils.slugify(getSubject()), 150) + "-" + user.getEmail() + "-" + minuteStamp + getEmailType(); } } // An email can then me sent as so: new ResetEmailEmailer(user, encryptedToken, returnUrl).sendEmail();

Additional stallion.toml Settings

There a few more optional settings to note:

[email]
# Defaults to true. If true, in non-prod mode, disables sending emails to
# anybody who is not match the white-list patterns. All emails instead will
# go to the outboundEmailTestAddress
restrictOutboundEmails = true
# A list of regular expressions defining which email addresses are allowed in
# non-prod mode
allowedTestingOutboundEmailPatterns = [".*@stallion.io"]
# List of email addresses that are allowed in non-prod mode
allowedOutboundEmails = ["test@stallion.io"]
# A test email address that all non-whitelisted email addresses will be routed to.
outboundEmailTestAddress = "test@stallion.io"
© 2025 Stallion Software LLC