-
Notifications
You must be signed in to change notification settings - Fork 18
/
concurrent_sender.cr
157 lines (145 loc) · 5.13 KB
/
concurrent_sender.cr
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
# Utility object for concurrent email sending.
#
# ```crystal
# rcpt_list = ["a@example.com", "b@example.com", "c@example.com", "d@example.com"]
#
# # Set SMTP client configuration
# config = EMail::Client::Config.new("your.mx.example.com", 25)
#
# # Create concurrent sender object
# sender = EMail::ConcurrentSender.new(config)
#
# # Sending emails with concurrently 3 connections.
# sender.number_of_connections = 3
#
# # Sending max 10 emails by 1 connection.
# sender.messages_per_connection = 10
#
# # Start email sending.
# sender.start do
# # In this block, default receiver is sender
# rcpts_list.each do |rcpt_to|
# # Create email message
# mail = EMail::Message.new
# mail.from "your_addr@example.com"
# mail.to rcpt_to
# mail.subject "Concurrent email sending"
# mail.message "message to #{rcpt_to}"
# # Enqueue the email to sender
# enqueue mail
# end
# end
# ```
class EMail::ConcurrentSender
@config : EMail::Client::Config
@queue : Array(Message) = Array(Message).new
@connections : Array(Fiber) = Array(Fiber).new
@connection_count : Int32 = 0
@started : Bool = false
@finished : Bool = false
@number_of_connections : Int32 = 1
@messages_per_connection : Int32 = 10
@connection_interval : Int32 = 200
# Create sender object with given client settings as EMail::Client::Config object.
def initialize(@config)
end
# Send one email with given client settings as several arguments.
#
# Avairable arguments are same as `EMail::Client::Conifg.create` method.
def initialize(*args, **named_args)
initialize(EMail::Client::Config.create(*args, **named_args))
end
# Set the maximum number of SMTP connections established at the same time.
def number_of_connections=(new_value : Int32)
raise EMail::Error::SenderError.new("Parameters cannot be set after start sending") if @started
raise EMail::Error::SenderError.new("Number of connections must be 1 or greater(given: #{new_value})") if new_value < 1
@number_of_connections = new_value
end
# Set the maximum number of email messages sent by one SMTP connection.
#
# When the number of sent emails by some SMTP connection reaches this parameter, the current connection will be closed and new one will be opened.
def messages_per_connection=(new_value : Int32)
raise EMail::Error::SenderError.new("Parameters cannot be set after start sending") if @started
raise EMail::Error::SenderError.new("Messages per connection must be 1 or greater(given: #{new_value})") if new_value < 1
@messages_per_connection = new_value
end
# Set the interval milliseconds between some connection is closed and new one is opened.
def connection_interval=(new_interval : Int32)
raise EMail::Error::SenderError.new("Parameters cannot be set after start sending") if @started
raise EMail::Error::SenderError.new("Connection interval must be 0 or greater(given: #{new_interval})") if new_interval < 0
@connection_interval = new_interval
end
# Enqueue a email message.
def enqueue(message : Message)
@queue << message.validate!
Fiber.yield
end
# Encueue email messages at the same time.
def enqueue(messages : Array(Message))
messages.each do |message|
enqueue(message)
end
end
# Starts sending emails.
#
# In the block of this method, the default receiver is `self`
def start
raise EMail::Error::SenderError.new("Email sending is already started") if @started
@started = true
spawn_sender
with self yield
@finished = true
until @queue.empty? && @connections.empty?
Fiber.yield
end
@started = false
@finished = false
end
# Starts sending emails with given parameters.
def start(number_of_connections : Int32? = nil, messages_per_connection : Int32? = nil, connection_interval : Int32? = nil)
raise EMail::Error::SenderError.new("Email sending is already started") if @started
self.number_of_connections = number_of_connections if number_of_connections
self.messages_per_connection = messages_per_connection if messages_per_connection
self.connection_interval = connection_interval if connection_interval
@started = true
spawn_sender
with self yield
@finished = true
until @queue.empty? && @connections.empty?
Fiber.yield
end
end
private def spawn_sender
spawn do
until @finished && @queue.empty?
spawn_client if !@queue.empty? && @connections.size < @number_of_connections
Fiber.yield
end
end
end
private def spawn_client
spawn do
@connections << Fiber.current
message = @queue.shift?
while message
@connection_count += 1
client = Client.new(@config)
client.number = @connection_count
client.start do
sent_messages = 0
while message && sent_messages < @messages_per_connection
send(message)
sent_messages += 1
Fiber.yield
message = @queue.shift?
end
end
sleep(@connection_interval.milliseconds) if @connection_interval > 0
end
@connections.delete(Fiber.current)
end
end
end
module EMail
alias Sender = EMail::ConcurrentSender
end