Skip to content

Latest commit

 

History

History
51 lines (37 loc) · 3.5 KB

README.md

File metadata and controls

51 lines (37 loc) · 3.5 KB

go-sender

go-sender is an event-based microservice that sends basic emails through various email providers (sendgrid, sparkpost). If one of the services goes down, go-sender will quickly failover to the secondary configuable email provider. It consumes off rabbitmq for email events and hands the event to a pool of email workers that calls out to the apppropriate email provider.

Running locally

To Run (replace docker-compose.yml with appropriate keys):

docker-compose build && docker-compose up

Unit Test:

go test ./... -v

Integration test

docker-compose build && docker-compose up -d
go test -tags=integration -sgkey=sgAPIKey -spkey=spAPIKey

Email Request Event

Users of this service will need to send this email event payload to rabbitmq. amqp://guest:guest@rabbitmq:5672/ (local development),

Alternatively, for testing purposes, one can submit the payload through the rabbitmq management page (http://localhost:15672/#/) (Username: guest, Password: guest) **

```
{
    "toEmail": "[email protected]",
    "fromEmail": "[email protected]",
    "subject": "[email protected]",
    "text": "sometext"
}
```

External Dependencies

  1. RabbitMQ - consumes/publishes messages to emailer.incoming.queue, emailer.retry.queue, emailer.errors.queue. RabbitMQ configs and definitions are located in the /scripts folder. Volumed onto the RabbitMQ image in docker-compose.yml.
  2. SendGrid - primary email provider
  3. Sparkpost - secondary email provider

Architecture

This email system consumes from two queues (emailer.incoming.queue) and emailer.retry.queue. One for the primary email provider (sendgrid) and another for the secondary (sparkpost). I defined failure as either the http request failing due to network issues or if the http status code returns anything above 500. If SendGrid were to go down, email events would be routed to the retry queue where Sparkpost will attempt to process the event. If SparkPost is also down, the event gets routed to the emailer.errors.queue where they have a TTL of 24 hours before they get discarded. If I had additional time, I would have like to have metrics and alerting around the rabbitmq queue size, status code/responses times from email providers,

An event base architecture for sending out email has multiple advantages in terms of scale and reliability. Events could be easily be replayed in case of downtime from external APIs. Initially I had written this service as an HTTP API with a circuit breaker than fell back to a secondary email provider. While this was fine, it also poses the issue if both email providers were down. Essentially we'd have to store those failed events to be retried somewhere. Storing these request in RabbitMQ or another event store allows us to replay events without much difficulty.

This email system has its limitations in that end users would know not necessarily know if their request failed/succeeded. I feel like in most email systems, emails are sent asynchronously. Given time restrictions, it would like to:

  1. Fix logging in the application, would remove the standard logger with something more verbose
  2. Basic request validation
  3. Metrics around status codes/response times, errors, and rabbitmq queue size

Additional Notes

  • Sparkpost does not allow you send from a non whitelisted domain. Their free account only lets you send from a sandbox account setting up to 50 emails. I used all 50 calls.
  • Email providers sometimes have delays in sending emails on free accounts. test