|
| 1 | +#!/usr/bin/env ruby |
| 2 | + |
| 3 | +require "bunny" |
| 4 | + |
| 5 | +MESSAGE_SIZES = [ |
| 6 | + ["12 B", 12, 100_000], |
| 7 | + ["1 KiB", 1_024, 100_000], |
| 8 | + ["4 KiB", 4_096, 100_000], |
| 9 | + ["16 KiB", 16_384, 100_000], |
| 10 | +] |
| 11 | + |
| 12 | +PREFETCH = 500 |
| 13 | +MULTI_ACK = 100 |
| 14 | +BATCH_SIZE = 500 |
| 15 | + |
| 16 | +puts |
| 17 | +puts "-" * 72 |
| 18 | +puts "Bunny #{Bunny::VERSION} on #{RUBY_DESCRIPTION}" |
| 19 | +puts "-" * 72 |
| 20 | + |
| 21 | +def run_benchmark(label, size, count, use_batch: false) |
| 22 | + payload = "x" * size |
| 23 | + |
| 24 | + pub_conn = Bunny.new |
| 25 | + pub_conn.start |
| 26 | + con_conn = Bunny.new |
| 27 | + con_conn.start |
| 28 | + |
| 29 | + pub_ch = pub_conn.create_channel |
| 30 | + con_ch = con_conn.create_channel |
| 31 | + con_ch.prefetch(PREFETCH) |
| 32 | + |
| 33 | + q = con_ch.queue("bunny.bench", auto_delete: true) |
| 34 | + q.purge |
| 35 | + |
| 36 | + received = 0 |
| 37 | + done = Queue.new |
| 38 | + |
| 39 | + q.subscribe(manual_ack: true) do |delivery_info, _properties, _body| |
| 40 | + received += 1 |
| 41 | + if (received % MULTI_ACK).zero? |
| 42 | + con_ch.basic_ack(delivery_info.delivery_tag, true) |
| 43 | + end |
| 44 | + done.push(true) if received == count |
| 45 | + end |
| 46 | + |
| 47 | + t0 = Process.clock_gettime(Process::CLOCK_MONOTONIC) |
| 48 | + |
| 49 | + if use_batch |
| 50 | + payloads = Array.new(BATCH_SIZE, payload) |
| 51 | + (count / BATCH_SIZE).times { pub_ch.basic_publish_batch(payloads, "", q.name, persistent: false) } |
| 52 | + remainder = count % BATCH_SIZE |
| 53 | + if remainder > 0 |
| 54 | + pub_ch.basic_publish_batch(Array.new(remainder, payload), "", q.name, persistent: false) |
| 55 | + end |
| 56 | + else |
| 57 | + count.times { pub_ch.basic_publish(payload, "", q.name, persistent: false) } |
| 58 | + end |
| 59 | + done.pop |
| 60 | + |
| 61 | + elapsed = Process.clock_gettime(Process::CLOCK_MONOTONIC) - t0 |
| 62 | + rate = (count / elapsed).round(0) |
| 63 | + mb_sec = (count * size / elapsed / 1_048_576.0).round(1) |
| 64 | + |
| 65 | + pub_conn.close |
| 66 | + con_conn.close |
| 67 | + |
| 68 | + [label, count, rate, mb_sec] |
| 69 | +end |
| 70 | + |
| 71 | +puts |
| 72 | +puts "## basic_publish (per-message)" |
| 73 | +puts |
| 74 | + |
| 75 | +results = [] |
| 76 | +MESSAGE_SIZES.each do |label, size, count| |
| 77 | + r = run_benchmark(label, size, count) |
| 78 | + results << r |
| 79 | + printf "%-12s %6dK msgs %8d msg/sec %8.1f MB/sec\n", r[0], r[1] / 1000, r[2], r[3] |
| 80 | +end |
| 81 | + |
| 82 | +puts |
| 83 | +puts "## basic_publish_batch (#{BATCH_SIZE} msgs/batch)" |
| 84 | +puts |
| 85 | + |
| 86 | +batch_results = [] |
| 87 | +MESSAGE_SIZES.each do |label, size, count| |
| 88 | + r = run_benchmark(label, size, count, use_batch: true) |
| 89 | + batch_results << r |
| 90 | + printf "%-12s %6dK msgs %8d msg/sec %8.1f MB/sec\n", r[0], r[1] / 1000, r[2], r[3] |
| 91 | +end |
| 92 | + |
| 93 | +puts |
| 94 | +puts "-" * 72 |
| 95 | +puts "| Workload | basic_publish | batch (#{BATCH_SIZE}) |" |
| 96 | +puts "|----------|--------:|-------:|" |
| 97 | +results.zip(batch_results).each do |r, br| |
| 98 | + puts "| #{r[1] / 1000}K x #{r[0]} | #{r[2]} msg/sec | #{br[2]} msg/sec |" |
| 99 | +end |
| 100 | +puts "-" * 72 |
0 commit comments