@@ -153,6 +153,50 @@ workflow = Workflow(
153
153
In this example, Task 2 will run roughly 1 second after Task 1 finishes, and
154
154
Task 3 and will run 2 seconds after Task 2 finishes.
155
155
156
+ ### Large Workflows
157
+
158
+ Because of how ` dramatiq-workflow ` is implemented, each task in a workflow has
159
+ to know about the remaining tasks in the workflow that could potentially run
160
+ after it. When a workflow has a large number of tasks, this can lead to an
161
+ increase of memory usage in the broker and increased network traffic between
162
+ the broker and the workers, especially when using ` Group ` tasks: Each task in a
163
+ ` Group ` can potentially be the last one to finish, so each task has to retain a
164
+ copy of the remaining tasks that run after the ` Group ` .
165
+
166
+ There are a few things you can do to alleviate this issue:
167
+
168
+ - Minimize the usage of parameters in the ` message ` method. Instead, consider
169
+ using a database to store data that is required by your tasks.
170
+ - Limit the size of groups to a reasonable number of tasks. Instead of
171
+ scheduling one task with 1000 tasks in a group, consider scheduling 10 groups
172
+ with 100 tasks each and chaining them together.
173
+ - Consider breaking down large workflows into smaller partial workflows that
174
+ then schedule a subsequent workflow at the very end of the outermost ` Chain ` .
175
+
176
+ Lastly, you can use compression to reduce the size of the messages in your
177
+ queue. While dramatiq does not provide a compression implementation by default,
178
+ one can be added with just a few lines of code. For example:
179
+
180
+ ``` python
181
+ import dramatiq
182
+ from dramatiq.encoder import JSONEncoder, MessageData
183
+ import lz4.frame
184
+
185
+ class DramatiqLz4JSONEncoder (JSONEncoder ):
186
+ def encode (self , data : MessageData) -> bytes :
187
+ return lz4.frame.compress(super ().encode(data))
188
+
189
+ def decode (self , data : bytes ) -> MessageData:
190
+ try :
191
+ decompressed = lz4.frame.decompress(data)
192
+ except RuntimeError :
193
+ # Uncompressed data from before the switch to lz4
194
+ decompressed = data
195
+ return super ().decode(decompressed)
196
+
197
+ dramatiq.set_encoder(DramatiqLz4JSONEncoder())
198
+ ```
199
+
156
200
## License
157
201
158
202
This project is licensed under the MIT License. See the [ LICENSE] ( LICENSE ) file
0 commit comments