Class: DBus::Connection

Inherits:
Object
  • Object
show all
Defined in:
lib/dbus/bus.rb

Overview

D-Bus main connection class

Main class that maintains a connection to a bus and can handle incoming and outgoing messages.

Direct Known Subclasses

SessionBus, SystemBus

Defined Under Namespace

Classes: NameRequestError

Constant Summary collapse

NAME_FLAG_ALLOW_REPLACEMENT =

FIXME: describe the following names, flags and constants. See DBus spec for definition

0x1
NAME_FLAG_REPLACE_EXISTING =
0x2
NAME_FLAG_DO_NOT_QUEUE =
0x4
REQUEST_NAME_REPLY_PRIMARY_OWNER =
0x1
REQUEST_NAME_REPLY_IN_QUEUE =
0x2
REQUEST_NAME_REPLY_EXISTS =
0x3
REQUEST_NAME_REPLY_ALREADY_OWNER =
0x4
DBUSXMLINTRO =
'<!DOCTYPE node PUBLIC "-//freedesktop//DTD D-BUS Object Introspection 1.0//EN"
"http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd">
<node>
  <interface name="org.freedesktop.DBus.Introspectable">
    <method name="Introspect">
      <arg name="data" direction="out" type="s"/>
    </method>
  </interface>
  <interface name="org.freedesktop.DBus">
    <method name="RequestName">
      <arg direction="in" type="s"/>
      <arg direction="in" type="u"/>
      <arg direction="out" type="u"/>
    </method>
    <method name="ReleaseName">
      <arg direction="in" type="s"/>
      <arg direction="out" type="u"/>
    </method>
    <method name="StartServiceByName">
      <arg direction="in" type="s"/>
      <arg direction="in" type="u"/>
      <arg direction="out" type="u"/>
    </method>
    <method name="Hello">
      <arg direction="out" type="s"/>
    </method>
    <method name="NameHasOwner">
      <arg direction="in" type="s"/>
      <arg direction="out" type="b"/>
    </method>
    <method name="ListNames">
      <arg direction="out" type="as"/>
    </method>
    <method name="ListActivatableNames">
      <arg direction="out" type="as"/>
    </method>
    <method name="AddMatch">
      <arg direction="in" type="s"/>
    </method>
    <method name="RemoveMatch">
      <arg direction="in" type="s"/>
    </method>
    <method name="GetNameOwner">
      <arg direction="in" type="s"/>
      <arg direction="out" type="s"/>
    </method>
    <method name="ListQueuedOwners">
      <arg direction="in" type="s"/>
      <arg direction="out" type="as"/>
    </method>
    <method name="GetConnectionUnixUser">
      <arg direction="in" type="s"/>
      <arg direction="out" type="u"/>
    </method>
    <method name="GetConnectionUnixProcessID">
      <arg direction="in" type="s"/>
      <arg direction="out" type="u"/>
    </method>
    <method name="GetConnectionSELinuxSecurityContext">
      <arg direction="in" type="s"/>
      <arg direction="out" type="ay"/>
    </method>
    <method name="ReloadConfig">
    </method>
    <signal name="NameOwnerChanged">
      <arg type="s"/>
      <arg type="s"/>
      <arg type="s"/>
    </signal>
    <signal name="NameLost">
      <arg type="s"/>
    </signal>
    <signal name="NameAcquired">
      <arg type="s"/>
    </signal>
  </interface>
</node>
'
MSG_BUF_SIZE =

The buffer size for messages.

4096

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(path) ⇒ Connection

and is something like: “transport1:key1=value1,key2=value2;transport2:key1=value1,key2=value2” e.g. “unix:path=/tmp/dbus-test” or “tcp:host=localhost,port=2687”



178
179
180
181
182
183
184
185
186
187
# File 'lib/dbus/bus.rb', line 178

def initialize(path)
  @path = path
  @unique_name = nil
  @buffer = ""
  @method_call_replies = Hash.new
  @method_call_msgs = Hash.new
  @signal_matchrules = Array.new
  @proxy = nil
  @object_root = Node.new("/")
end

Instance Attribute Details

#socketObject (readonly)

The socket that is used to connect with the bus.



170
171
172
# File 'lib/dbus/bus.rb', line 170

def socket
  @socket
end

#unique_nameObject (readonly)

The unique name (by specification) of the message.



168
169
170
# File 'lib/dbus/bus.rb', line 168

def unique_name
  @unique_name
end

Instance Method Details

#add_match(mr, &slot) ⇒ Object

Asks bus to send us messages matching mr, and execute slot when received



540
541
542
543
544
# File 'lib/dbus/bus.rb', line 540

def add_match(mr, &slot)
  # check this is a signal.
  @signal_matchrules << [mr, slot]
  self.proxy.AddMatch(mr.to_s)
end

#connectObject

Connect to the bus and initialize the connection.



190
191
192
193
# File 'lib/dbus/bus.rb', line 190

def connect
  connect_to_tcp if @path.include? "tcp:" #connect to tcp socket
  connect_to_unix if @path.include? "unix:" #connect to unix socket
end

#connect_to_tcpObject

Connect to a bus over tcp and initialize the connection.



196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
# File 'lib/dbus/bus.rb', line 196

def connect_to_tcp
  #check if the path is sufficient
  if @path.include? "host=" and @path.include? "port="
    host,port,family = "","",""
    #get the parameters
    @path.split(",").each do |para|
      host = para.sub("tcp:","").sub("host=","") if para.include? "host="
      port = para.sub("port=","").to_i if para.include? "port="
      family = para.sub("family=","") if para.include? "family="
    end
    dlog "host,port,family : #{host},#{port},#{family}"      
    begin
      #initialize the tcp socket
      @socket = TCPSocket.new(host,port)
      init_connection
    rescue
      elog "Could not establish connection to: #{@path}, will now exit."
      exit(0) #a little harsh
    end
  else
    #Danger, Will Robinson: the specified "path" is not usable
    elog "supplied path: #{@path}, unusable! sorry."
  end
end

#connect_to_unixObject

Connect to an abstract unix bus and initialize the connection.



222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
# File 'lib/dbus/bus.rb', line 222

def connect_to_unix
  @socket = Socket.new(Socket::Constants::PF_UNIX,Socket::Constants::SOCK_STREAM, 0)
  parse_session_string
  if @transport == "unix" and @type == "abstract"
    if HOST_END == LIL_END
      sockaddr = "\1\0\0#{@unix_abstract}"
    else
      sockaddr = "\0\1\0#{@unix_abstract}"
    end
  elsif @transport == "unix" and @type == "path"
    sockaddr = Socket.pack_sockaddr_un(@unix)
  end
  @socket.connect(sockaddr)
  init_connection
end

#emit(service, obj, intf, sig, *args) ⇒ Object

Emit a signal event for the given service, object obj, interface intf and signal sig with arguments args.



608
609
610
611
612
613
614
615
616
617
618
619
620
# File 'lib/dbus/bus.rb', line 608

def emit(service, obj, intf, sig, *args)
  m = Message.new(DBus::Message::SIGNAL)
  m.path = obj.path
  m.interface = intf.name
  m.member = sig.name
  m.sender = service.name
  i = 0
  sig.params.each do |par|
    m.add_param(par[1], args[i])
    i += 1
  end
  send(m.marshall)
end

#glibizeObject

Tell a bus to register itself on the glib main loop



266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
# File 'lib/dbus/bus.rb', line 266

def glibize
  require 'glib2'
  # Circumvent a ruby-glib bug
  @channels ||= Array.new

  gio = GLib::IOChannel.new(@socket.fileno)
  @channels << gio
  gio.add_watch(GLib::IOChannel::IN) do |c, ch|
    update_buffer
    messages.each do |msg|
      process(msg)
    end
    true
  end
end

#introspect(dest, path) ⇒ Object

Issues a call to the org.freedesktop.DBus.Introspectable.Introspect method dest is the service and path the object path you want to introspect If a code block is given, the introspect call in asynchronous. If not data is returned

FIXME: link to ProxyObject data definition The returned object is a ProxyObject that has methods you can call to issue somme METHOD_CALL messages, and to setup to receive METHOD_RETURN



409
410
411
412
413
414
415
416
417
418
419
420
# File 'lib/dbus/bus.rb', line 409

def introspect(dest, path)
  if not block_given?
    # introspect in synchronous !
    data = introspect_data(dest, path)
    pof = DBus::ProxyObjectFactory.new(data, self, dest, path)
    return pof.build
  else
    introspect_data(dest, path) do |data|
      yield(DBus::ProxyObjectFactory.new(data, self, dest, path).build)
    end
  end
end

#introspect_data(dest, path) ⇒ Object



372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
# File 'lib/dbus/bus.rb', line 372

def introspect_data(dest, path)
  m = DBus::Message.new(DBus::Message::METHOD_CALL)
  m.path = path
  m.interface = "org.freedesktop.DBus.Introspectable"
  m.destination = dest
  m.member = "Introspect"
  m.sender = unique_name
  if not block_given?
    # introspect in synchronous !
    send_sync(m) do |rmsg|
      if rmsg.is_a?(Error)
        raise rmsg
      else
        return rmsg.params[0]
      end
    end
  else
    send(m.marshall)
    on_return(m) do |rmsg|
      if rmsg.is_a?(Error)
        yield rmsg
      else
        yield rmsg.params[0]
      end
    end
  end
  nil
end

#messagesObject

Retrieve all the messages that are currently in the buffer.



466
467
468
469
470
471
472
# File 'lib/dbus/bus.rb', line 466

def messages
  ret = Array.new
  while msg = pop_message
    ret << msg
  end
  ret
end

#on_return(m, &retc) ⇒ Object

Specify a code block that has to be executed when a reply for message m is received.



528
529
530
531
532
533
534
535
536
# File 'lib/dbus/bus.rb', line 528

def on_return(m, &retc)
  # Have a better exception here
  if m.message_type != Message::METHOD_CALL
    elog "Funky exception, occured."
    raise "on_return should only get method_calls"
  end
  @method_call_msgs[m.serial] = m
  @method_call_replies[m.serial] = retc
end

#parse_session_stringObject

Parse the session string (socket address).



239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
# File 'lib/dbus/bus.rb', line 239

def parse_session_string
  path_parsed = /^([^:]*):([^;]*)$/.match(@path)
  @transport = path_parsed[1]
  adr = path_parsed[2]
  if @transport == "unix"
    adr.split(",").each do |eqstr|
      idx, val = eqstr.split("=")
      case idx
      when "path"
        @type = idx
        @unix = val
      when "abstract"
        @type = idx
        @unix_abstract = val
      when "guid"
        @guid = val
      end
    end
  end
end

#poll_messagesObject

Update the buffer and retrieve all messages using Connection#messages. Return the messages.



479
480
481
482
483
484
485
486
# File 'lib/dbus/bus.rb', line 479

def poll_messages
  ret = nil
  r, d, d = IO.select([@socket], nil, nil, 0)
  if r and r.size > 0
    update_buffer
  end
  messages
end

#pop_messageObject

Get one message from the bus and remove it from the buffer. Return the message.



454
455
456
457
458
459
460
461
462
463
# File 'lib/dbus/bus.rb', line 454

def pop_message
  ret = nil
  begin
    ret, size = Message.new.unmarshall_buffer(@buffer)
    @buffer.slice!(0, size)
  rescue IncompleteBufferException => e
    # fall through, let ret be null
  end
  ret
end

#process(m) ⇒ Object

Process a message m based on its type.



547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
# File 'lib/dbus/bus.rb', line 547

def process(m)
  return if m.nil? #check if somethings wrong
  case m.message_type
  when Message::ERROR, Message::METHOD_RETURN
    raise InvalidPacketException if m.reply_serial == nil
    mcs = @method_call_replies[m.reply_serial]
    if not mcs
      dlog "no return code for mcs: #{mcs.inspect} m: #{m.inspect}"
    else
      if m.message_type == Message::ERROR
        mcs.call(Error.new(m))
      else
        mcs.call(m)
      end
      @method_call_replies.delete(m.reply_serial)
      @method_call_msgs.delete(m.reply_serial)
    end
  when DBus::Message::METHOD_CALL
    if m.path == "/org/freedesktop/DBus"
      dlog "Got method call on /org/freedesktop/DBus"
    end
    # handle introspectable as an exception:
    if m.interface == "org.freedesktop.DBus.Introspectable" and
        m.member == "Introspect"
      reply = Message.new(Message::METHOD_RETURN).reply_to(m)
      reply.sender = @unique_name
      node = @service.get_node(m.path)
      raise NotImplementedError if not node
      reply.sender = @unique_name
      reply.add_param(Type::STRING, @service.get_node(m.path).to_xml)
      send(reply.marshall)
    else
      node = @service.get_node(m.path)
      return if node.nil?
      obj = node.object
      return if obj.nil?
      obj.dispatch(m) if obj
    end
  when DBus::Message::SIGNAL
    @signal_matchrules.each do |elem|
      mr, slot = elem
      if mr.match(m)
        slot.call(m)
        return
      end
    end
  else
    dlog "Unknown message type: #{m.message_type}"
  end
end

#proxyObject

Set up a ProxyObject for the bus itself, since the bus is introspectable. Returns the object.



436
437
438
439
440
441
442
443
444
# File 'lib/dbus/bus.rb', line 436

def proxy
  if @proxy == nil
    path = "/org/freedesktop/DBus"
    dest = "org.freedesktop.DBus"
    pof = DBus::ProxyObjectFactory.new(DBUSXMLINTRO, self, dest, path)
    @proxy = pof.build["org.freedesktop.DBus"]
  end
  @proxy
end

#request_service(name) ⇒ Object

Attempt to request a service name.

Raises:



427
428
429
430
431
432
# File 'lib/dbus/bus.rb', line 427

def request_service(name)
  r = proxy.RequestName(name, NAME_FLAG_REPLACE_EXISTING)
  raise NameRequestError if r[0] != REQUEST_NAME_REPLY_PRIMARY_OWNER
  @service = Service.new(name, self)
  @service
end

#send(buf) ⇒ Object

Send the buffer buf to the bus using Connection#writel.



261
262
263
# File 'lib/dbus/bus.rb', line 261

def send(buf)
  @socket.write(buf) unless @socket.nil?
end

#send_sync(m, &retc) ⇒ Object

Send a message m on to the bus. This is done synchronously, thus the call will block until a reply message arrives.



507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
# File 'lib/dbus/bus.rb', line 507

def send_sync(m, &retc) # :yields: reply/return message
  return if m.nil? #check if somethings wrong
  send(m.marshall)
  @method_call_msgs[m.serial] = m
  @method_call_replies[m.serial] = retc

  retm = wait_for_message
  
  return if retm.nil? #check if somethings wrong
  
  process(retm)
  until [DBus::Message::ERROR,
      DBus::Message::METHOD_RETURN].include?(retm.message_type) and
      retm.reply_serial == m.serial
    retm = wait_for_message
    process(retm)
  end
end

#service(name) ⇒ Object Also known as: []

Retrieves the service with the given name.



599
600
601
602
603
# File 'lib/dbus/bus.rb', line 599

def service(name)
  # The service might not exist at this time so we cannot really check
  # anything
  Service.new(name, self)
end

#update_bufferObject

Fill (append) the buffer from data that might be available on the socket.



448
449
450
# File 'lib/dbus/bus.rb', line 448

def update_buffer
  @buffer += @socket.read_nonblock(MSG_BUF_SIZE) unless @socket.nil?
end

#wait_for_messageObject

Wait for a message to arrive. Return it once it is available.



489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
# File 'lib/dbus/bus.rb', line 489

def wait_for_message
  if @socket.nil?
    elog "Can't wait for messages, @socket is nil."
    return
  end
  ret = pop_message
  while ret == nil
    r, d, d = IO.select([@socket])
    if r and r[0] == @socket
      update_buffer
      ret = pop_message
    end
  end
  ret
end