Class: NatsMessaging::NatsService

Inherits:
Object
  • Object
show all
Defined in:
lib/nats_messaging/nats_service.rb

Instance Method Summary collapse

Constructor Details

#initialize(nats_url, stream_name: nil, subjects: []) ⇒ NatsService

Returns a new instance of NatsService.



6
7
8
9
10
11
12
13
# File 'lib/nats_messaging/nats_service.rb', line 6

def initialize(nats_url, stream_name: nil, subjects: [])
  @nats = NATS.connect(nats_url)
  if stream_name
    @js = @nats.jetstream
    create_stream(stream_name, subjects)
  end
  @subscriptions = {} # Store active subscriptions
end

Instance Method Details

#create_stream(stream_name, subjects) ⇒ Object

Create a stream



16
17
18
19
20
21
22
23
24
25
26
# File 'lib/nats_messaging/nats_service.rb', line 16

def create_stream(stream_name, subjects)
  begin
    stream = @js.stream_info(stream_name)
    puts "NATS: Stream #{stream_name} already exists"
  rescue NATS::JetStream::Error::NotFound
    @js.add_stream(name: stream_name, subjects: subjects)
    puts "NATS: Stream #{stream_name} created"
  rescue => e
    puts "NATS: Failed to create stream: #{e.message}"
  end
end

#listen_and_reply(subject, reply_message) ⇒ Object

Listen to a subject and reply with a message using MessagePack



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
# File 'lib/nats_messaging/nats_service.rb', line 101

def listen_and_reply(subject, reply_message)
  # Cancel active subscription if it already exists for this subject
  if @subscriptions[subject]
    @subscriptions[subject].unsubscribe
  end
  unpacked_data = nil
  begin
    # Create a new subscription
    puts "NATS: Listening on #{subject} with reply message: #{reply_message}"
    @subscriptions[subject] = @nats.subscribe(subject) do |msg|
      begin
        unpacked_data = MessagePack.unpack(msg.data) # Deserializar mensaje recibido
        puts "NATS: Received request on #{subject}: #{unpacked_data}"
        packed_reply = reply_message.to_msgpack # Serializar respuesta
        # Asegurar que msg.reply existe antes de responder
        if msg.reply
          @nats.publish(msg.reply, packed_reply)
          puts "NATS: Replied to #{subject} with message: #{reply_message}"
        else
          puts "NATS: No reply subject for #{subject}, cannot respond"
        end
        yield(msg.subject, unpacked_data) if block_given?
        unpacked_data
      rescue => e
        puts "NATS: Error while processing message: #{e.message}"
      end
    end
  rescue => e
    puts "NATS: Error while replying: #{e.message}"
  end
end

#process_received_message(msg, subject) ⇒ Object

Process received message



71
72
73
74
75
76
77
78
79
# File 'lib/nats_messaging/nats_service.rb', line 71

def process_received_message(msg, subject)
  begin
    unpacked_data = MessagePack.unpack(msg.data)
    puts "NATS: Message received on #{subject}: #{unpacked_data}"
    unpacked_data
  rescue => e
    puts "NATS: Error while processing message: #{e.message}"
  end
end

#publish_message(subject, message) ⇒ Object

Publish a message to a subject using MessagePack



29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
# File 'lib/nats_messaging/nats_service.rb', line 29

def publish_message(subject, message)
  packed_message = message.to_msgpack # Serializar a MessagePack

  if @js
    begin
      ack = @js.publish(subject, packed_message)
      puts "NATS: Message sent to #{subject}: #{message}, ACK: #{ack.inspect}"
    rescue NATS::JetStream::Error::NotFound
      @nats.publish(subject, packed_message)
      puts "NATS: Stream not found, falling back to regular NATS publish"
    rescue => e
      puts "NATS: Unexpected error: #{e.message}"
    end
  else
    begin
      @nats.publish(subject, packed_message)
      puts "NATS: Message sent to #{subject}: #{message}, ACK: #{ack.inspect}"
    rescue => e
      puts "NATS: Unexpected error: #{e.message}"
    end
  end
end

#send_request(subject, message) ⇒ Object

Send a request to a subject and wait for response using MessagePack



82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
# File 'lib/nats_messaging/nats_service.rb', line 82

def send_request(subject, message)
  puts "NATS: Sending request to #{subject} with message: #{message}"
  begin
    packed_message = message.to_msgpack # Serializar el mensaje
    response = @nats.request(subject, packed_message, timeout: 5) # Timeout of 5 seconds
    unpacked_response = MessagePack.unpack(response.data) # Deserializar respuesta
    puts "NATS: Received reply: #{unpacked_response}"
    yield(subject, unpacked_response) if block_given?
    unpacked_response
  rescue NATS::IO::Timeout
    puts "NATS: Request timed out for subject: #{subject}"
    nil
  rescue => e
    puts "NATS: Unexpected error: #{e.message}"
    nil
  end
end

#subscribe_to_subject(subject, durable_name = "durable_name") ⇒ Object

Subscribe to a subject using MessagePack



53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
# File 'lib/nats_messaging/nats_service.rb', line 53

def subscribe_to_subject(subject, durable_name = "durable_name")
  puts "NATS: Subscribing to #{subject}"
  if @js
    @subscriptions[subject] = @js.subscribe(subject, durable: durable_name) do |msg|
                    unpacked_data = process_received_message(msg, subject)
                    yield(msg.subject, unpacked_data) if block_given?
                    unpacked_data
                   end
  else
    @subscriptions[subject] = @nats.subscribe(subject) do |msg|
                    unpacked_data = process_received_message(msg, subject)
                    yield(msg.subject, unpacked_data) if block_given?
                    unpacked_data
                   end
  end
end