- Events
Black History Month in Tech: 7 Visionaries Who Shaped The Future
A few months ago we released automatic IP Warm Up, but we never got to talk about how it’s implemented. Today, we’re going to peek under the hood and try to understand what makes our IP warm up tick. We’re going to start with some context, and then we’ll dive into the interesting technical details later in the post.
Let’s face it — no one likes receiving tons of information all at once. The same can be said for mailbox providers receiving your emails. Spammers usually blast out their campaigns over a short period of time, so the providers don’t like that behavior. To protect themselves from abuse, they’ll block messages that come from an IP that has never sent an email before. Some providers might still accept the messages, but those messages will go right to spam. That’s not good for anybody.
To get those messages into the inbox we better start delivering fewer of them and ramp up the volume over time to build up a reputation; that’s why it is called “warm up.” Think about warming up a car engine before flooring it or doing light exercises before lifting weights at a gym, it’s a bit like that.
When you sign up for a Mailgun account, you get a shared IP. A shared IP means it’s used by multiple customers, so it’s already warmed up and has a good reputation. You don’t have to worry about it unless another sender causes its reputation to go down. Now you want to add a dedicated IP, and we’ve talked about why you might want that as your sending volume grows. The dedicated IP that you get is fresh and has no reputation, hence mailbox providers know nothing about it. We don’t pre-warm them because we want your IP to be tied to what you’re sending and so you can build your own reputation over time. That will lead to better results. Does it mean you still have to send only a few messages from the start? Yes, we know that it’s not ideal.
That’s where automatic IP warm up comes in. When a dedicated IP gets added, the shared IP still remains on your account, so we can divert the traffic to it during the warm-up. Over time, we can increase the volume a little bit at a time to send more and more messages until the dedicated IP is fully warmed up and you’re ready to go.
A very important thing to remember, however, is that the warm-up process will not buy you great deliverability! The practices you incorporate as a sender is what drives deliverability so make sure you are taking responsibility for the messages you send out and the recipients that receive them. Otherwise, you’re just slowly deteriorating the IP reputation instead of instantly burning it!
We’ve found that mailbox providers typically care about both hourly and daily volumes. Each day the volume could be increased by a few percent. This is very handy because this compound effect leads to exponential growth and the ability to ramp up in a relatively small amount of steps. We call these steps “stages”. Stages make up a “warm-up plan”, that looks like this:
If we plot the daily amounts, we’ll get the following picture:
It consists of 14 stages and at each stage, we send a certain amount of messages from a dedicated IP. That volume was carefully determined by our deliverability team through research and is periodically updated when we collect new evidence. It turned out that sending a proper amount of messages is an interesting engineering problem and we’d like to share some details on the way we solved it.
Let’s dive in and take a closer look at the implementation of an algorithm that allows us to send a given amount of messages. Now that we have a shared and a dedicated IP together, we should think about how to balance the traffic between the two of them. The easiest way to do that is to implement a random load-balancing algorithm. Imagine that we flip a coin to determine which IP a message is sent from, with the shared IP being “heads” and the dedicated IP being “tails.” If we wanted to decrease the number of messages coming from the dedicated IP, we would change the probability of getting “tails” as our outcome. This would make our coin unfair, but it works to our advantage. Here is how that might look in golang:
1var rnd = rand.New(rand.NewSource(98)) 2var probabilityOfDedicated = 0.23if rnd.Float64() < probabilityOfDedicated {4 return "dedicated"5}6return "shared"
The probabilityOfDedicated
can take any float value from 0 to 1.0. When it’s 1.0 all messages go through the dedicated IP and when it’s 0 everything goes through the shared one. Setting it to 0.5 will give us equal odds of sending it via the dedicated IP or shared IP. It’s essentially a knob that controls the sending flow. Let’s see how we can maintain a rate of 350 messages an hour from the dedicated IP by updating that value. The simplest thing that comes to mind is setting the probabilityOfDedicated
to 1 and once the 350 message mark is reached, reset it to 0. It’s a little abrupt, but should do the trick.
1const cap = 3502func getProbabilityOfDedicated(sentMessages int) float64 {3 if sentMessages < cap {4 return 15 }6 return 07}
When we want to run a program on hundreds of servers processing hundreds of thousands of requests per second we can’t expect everything to be always updated in real-time. Optimization techniques such as batch processing and caching become our best friends. They’re great at what they were designed for, but they inevitably add a lag between a value being calculated, updated, and shared across all the nodes in a cluster. Let’s call that lag inertia from here on out.
Speaking of our example, there are many things that contribute to inertia. First, the statistics updates might not be truly real-time due to optimizations in the message counting service. Second, there is a lag between when our value of probability of picking dedicated IP gets calculated and distributed due to caching. By the time we see that a 350th message has been sent and it’s time to stop, more might slip through before the update takes an effect.
This is a working example, so feel free to paste it to go playground and play with it:
1package main23import (4 "fmt"5 "math/rand"6 "sync"7 "time"8)910var (11 mutex = &sync.Mutex{}12 rnd = rand.New(rand.NewSource(99))13 messagesSent = 0 // keep track of the messages that IP sends14 probability = 0.0 // probability of picking a dedicated IP15)1617const (18 sampleRateMsec = 50.0 // how often the system is being refreshed19 cap = 350.0 // this is how many messages we want to send20)2122// deliverySimulator will emulate the message sending process.23// It will pick the dedicated IP based on the probability.24func deliverySimulator() {25 for {26 mutex.Lock()27 if rnd.Float64() < probability {28 // We don't know how fast the messages are29 // being submitted, let's throw some30 // randomness in there to make it unpredictable.31 messagesSent += rnd.Intn(10)32 } else {33 // Shared IP will be picked here.34 }35 mutex.Unlock()36 time.Sleep(time.Millisecond)37 }38}3940func calcProbabilityOfDedicated(sent int) float64 {41 if sent < cap {42 return 143 }44 return 045}4647func main() {48 go deliverySimulator()4950 // Control loop.51 for i := 0; i < 8; i++ {52 mutex.Lock()5354 // Get messages.55 sent := messagesSent5657 // Calculate probability.58 probability = calcProbabilityOfDedicated(sent)5960 fmt.Println("sent:", sent, "probability:", probability)61 mutex.Unlock()6263 time.Sleep(time.Duration(sampleRateMsec) * time.Millisecond)64 }65}
Here is the output we get:
Oops, looks like we sent many more messages than we wanted; almost 25% more as a matter of fact. What can we do about it? Well, we can come up with a threshold to stop it in advance, but that would have to be different for every customer as they all submit messages at different rates.
What if we can embrace inertia? What if we could gradually slow down?
Inertia is so common in the physical world. In order to control mechanical systems, engineers have to always take inertia into account. Can we learn anything from them?
The first attempts of creating an automatic controller for physical systems date back to 17th century with the centrifugal governor. It was an essential part of any steam engine at the time. Fast forward to the early 20th century where the theory behind Proportional Integral Derivative (PID) controller was developed after observing how helmsmen control ships by taking into account different environmental and system factors. These days PID controllers are used everywhere from quadcopters to toasters, and it’s because they are very simple and effective. Of course, there are many other control algorithms, but most of them are not as nearly as elegant.
Let’s talk about the theory behind PID controllers. For starters, they only work when there is a feedback loop. Feedback loops are pretty straightforward in that the result of the control action gets piped back to the controller input. It’s like an ouroboros if the mythology symbolism helps, and if not, there’s this diagram.
+------------+
input | | output
+------> | | +------>
| system |
control | | feedback
+------> | | +------+
| +------------+ |
| |
| +------------+ |
+------+ | controller | <------+
+------------+
Diagram 1
Let’s take a closer look at the controller block and open it up.
Diagram 2
Whoa, that’s a lot. It’s a little overwhelming so let’s break it down a bit. First, we need to agree on terminology. Here are a few terms to be familiar with going in:
Process variable – y(t), the current value of the thing we want to control. In our case, it would be the number of messages sent from an IP and will serve as the feedback.
Setpoint – r(t), the value we want to land at. In our example, it’s the cap of 350 messages.
Error – e(t), the current difference between a setpoint and a process variable. e(t) = r(t) – y(t).
Control variable – u(t), the output of the controller, this will drive our probability of picking a dedicated IP.
All variables are written as a function f(t), which means its value changes in relation to time t, so it’s a value at that very moment of time. The diagram can be described by the classic formula from any textbook on the control theory:
Ok, let’s figure out what all that math means.
There are 4 terms in this equation — proportional + integral + derivative + bias. Each of them contribute to the control signal and the weight of their contributions can be set by respective coefficients: Kp, Ki, Kd and Kb.
Proportional term (block P of diagram 2) is the most important term of the equation. As the name states, it’s proportional to the error. The bigger the error, the bigger control force will be applied to the system.
Integral term (block I of diagram 2) is useful for getting to the setpoint faster. It accumulates the errors over time and tries to help the proportional term to correct for them. It has a very important drawback for our use case though — potential overshoot. The nature of the integral term is that it’s slow to kick in at the beginning, but when enough error is accumulated over time it can put the system in overdrive getting past the setpoint. In the most extreme case, it can lead to what is called integral windup.
Derivative term (block D of diagram 2) is proportional to the rate of change. As we reduce the error, the derivative term actually becomes negative. We can think of it as if we approach the setpoint too fast it, it will slow us down. Usually, it results to a more gradual change of the process variable.
Bias term (block B of diagram 2) is just a constant; it’s not used often but might be useful to still drive the value when it’s close to the setpoint. At that point, the error is small and the proportional term is not that pronounced.
Think about these terms as if they were all in a panel of judges voting on the decision of what the next step should be. The controller acts based on the scores they give. Since they have different personalities they all score a bit differently. The proportional is the most reasonable judge and always gives a higher score the further we are from the setpoint. The integral gets impatient if we’re not reducing the error for too long so it scores higher as time goes on. The derivative is always nervous and scores negatively if the error is reduced too fast. The bias always scores the same no matter what.
To get a feel on how the terms affect the controls there is a cool spaceship demo.
Now that we’re familiar with the theory, let’s apply it. We can use the existing control loop from the previous example. One of the hardest thing implementing this is to pick proper control coefficients. Tuning the PID process is not straightforward, and there are few techniques out there that can help us with that. However, we’ve found that the fastest way is to create a simulation script and play with the parameters. They might not be perfect, but they will be good enough to solve the problem.
Here is how we might think of the coefficients in regards to our problem.
Proportional will decrease the probability of picking a dedicated IP the closer we are to the desired cap. So, when we start, the term value is big and it goes down to 0 as we send more and more messages.
Integral is tricky and we might not even need it, as we can’t correct the overshoot. The messages can’t be deleted once they are sent out.
Derivative is a very important one. It will help us to correct for the situation when the messages are submitted too fast. When that happens the term becomes big enough and starts counteracting other terms. Since it is negative when that happens it will reduce the probability and, in turn, reduce the amount of messages that get sent out from a dedicated IP.
Bias might be very useful if we’re not planning to rely on the integral term. When we get really close to the desired cap the proportional term will be small and it won’t be able to drive the value toward it as hard. This is when bias could give it a little kick.
Now that we know all of that we can replace calcProbabilityOfDedicated
with the following code:
1const (2 // These coefficients has been picked by trial-and-error.3 Kp = 0.42 // proportional coefficient4 Ki = 0.000001 // can be removed to simplify the algo5 Kd = 1.2 // derivative coefficient6 Kb = 0.1 // fairly small in this example7)89var (10 errorVal = 0.011 prevError = 0.012 integral = 0.013)1415func calcProbabilityOfDedicated(sent int) float64 {16 prevError = errorVal1718 // Calculate terms.19 errorVal = cap - float64(sent)20 integral = integral + (errorVal * float64(sampleRateMsec))21 derivative := (errorVal - prevError) / float64(sampleRateMsec)2223 // Calculate PID control response24 probability := Kp*errorVal + Ki*integral + Kd*derivative + Kb2526 // With given parameters the output is roughly within [0: 100]27 // Let's shape it to [0: 1]28 probability /= 10029 if probability > 1 {30 return 131 }32 if probability < 0 {33 return 034 }3536 return probability37}
So we get this:
As you can see it works like a charm. We quickly ramped up to 300, slowed down and eventually got to 350, then the probability went down close to 0 naturally. How cool is that?!
In reality, the algorithm might not set the value precisely, but it has one major advantage — it can easily be distributed. Imagine if one of the workers picks up and locks an IP, calculates the probability, saves it, and then off to the next one. This allows us to control the volume of thousands of IPs simultaneously.
Now that we have that figured out, we can create a warm up plan. It would consist of stages, where each stage has a maximum amount of messages assigned to it. As we progress through the stages this maximum volume increases and you are free to send out more and more messages. This gives you the ability to send as fast as you’re used to from a shared IP, while we gradually warm up a fresh dedicated one.
Let’s see how it works in production. Here we plotted the volume at each stage. The desired values come from the plan and actual are what was sent out.
Looks good! Before rolling it out we created a simulation environment that allowed us to test different scenarios. This is how we managed to get the values for the coefficients lined up. To our big surprise, it worked out when we deployed it to production the first time… oh, if only it happened for every deployment!
Last updated on August 27, 2020
Black History Month in Tech: 7 Visionaries Who Shaped The Future
Happy Festivus: Email Deliverability Guide For The Holiday Season
Email's Best of 2020
Our Values In Action
How To Use Parallel Programming
Golang’s Superior Cache Solution to Memcached and Redis
HTTP/2 Cleartext (H2C) Client Example in Go
How we built a Lucene-inspired parser in Go
Gubernator: Cloud-native distributed rate limiting for microservices
Delivering HTML Emails With Mailgun-Go
InboxReady x Salesforce: The Key to a Stronger Email Deliverability
Become an Email Pro With Our Templates API
Google Postmaster Tools: Understanding Sender Reputation
Navigating Your Career as a Woman in Tech
Implementing Dmarc – A Step-by-Step Guide
Email Bounces: What To Do About Them
Announcing InboxReady: The deliverability suite you need to hit the inbox
Black History Month in Tech: 7 Visionaries Who Shaped The Future
How To Create a Successful Triggered Email Program
Designing HTML Email Templates For Transactional Emails
InboxReady x Salesforce: The Key to a Stronger Email Deliverability
Implementing Dmarc – A Step-by-Step Guide
Announcing InboxReady: The deliverability suite you need to hit the inbox
Designing HTML Email Templates For Transactional Emails
Email Security Best Practices: How To Keep Your Email Program Safe
Mailgun’s Active Defense Against Log4j
Email Blasts: The Dos And Many Don’ts Of Mass Email Sending
Email's Best of 2021
5 Ideas For Better Developer-Designer Collaboration
Mailgun Joins Sinch: The Future of Customer Communications Is Here
Always be in the know and grab free email resources!
By sending this form, I agree that Mailgun may contact me and process my data in accordance with its Privacy Policy.