7 import util.go_utils as gobject_utils
8 import util.misc as misc_utils
11 _moduleLogger = logging.getLogger(__name__)
15 tp.ChannelTypeStreamedMedia,
16 tp.ChannelInterfaceGroup,
17 tp.ChannelInterfaceCallState,
18 tp.ChannelInterfaceHold,
21 def __init__(self, connection, manager, props, contactHandle):
22 self.__manager = manager
24 self._delayedClose = gobject_utils.Timeout(self._on_close_requested)
26 if telepathy.interfaces.CHANNEL_INTERFACE + '.InitiatorHandle' in props:
27 self._initiator = connection.get_handle_by_id(
28 telepathy.HANDLE_TYPE_CONTACT,
29 props[telepathy.interfaces.CHANNEL_INTERFACE + '.InitiatorHandle'],
31 elif telepathy.interfaces.CHANNEL_INTERFACE + '.InitiatorID' in props:
32 self._initiator = connection.get_handle_by_name(
33 telepathy.HANDLE_TYPE_CONTACT,
34 props[telepathy.interfaces.CHANNEL_INTERFACE + '.InitiatorHandle'],
37 # Maemo 5 seems to require InitiatorHandle/InitiatorID to be set
38 # even though I can't find them in the dbus spec. I think its
39 # generally safe to assume that its locally initiated if not
40 # specified. Specially for The One Ring, its always locally
42 _moduleLogger.warning('InitiatorID or InitiatorHandle not set on new channel, assuming locally initiated')
43 self._initiator = connection.GetSelfHandle()
45 tp.ChannelTypeStreamedMedia.__init__(self, connection, manager, props)
46 tp.ChannelInterfaceGroup.__init__(self)
47 tp.ChannelInterfaceCallState.__init__(self)
48 tp.ChannelInterfaceHold.__init__(self)
49 self.__contactHandle = contactHandle
50 self.__calledNumber = None
52 self._implement_property_get(
53 telepathy.interfaces.CHANNEL_INTERFACE,
55 'InitiatorHandle': lambda: dbus.UInt32(self._initiator.id),
56 'InitiatorID': lambda: self._initiator.name,
59 self._add_immutables({
60 'InitiatorHandle': telepathy.interfaces.CHANNEL_INTERFACE,
61 'InitiatorID': telepathy.interfaces.CHANNEL_INTERFACE,
63 self._implement_property_get(
64 telepathy.interfaces.CHANNEL_INTERFACE_GROUP,
66 'LocalPendingMembers': lambda: self.GetLocalPendingMembersWithInfo()
69 self._implement_property_get(
70 telepathy.interfaces.CHANNEL_TYPE_STREAMED_MEDIA,
72 "InitialAudio": self.initial_audio,
73 "InitialVideo": self.initial_video,
76 self._add_immutables({
77 'InitialAudio': telepathy.interfaces.CHANNEL_TYPE_STREAMED_MEDIA,
78 'InitialVideo': telepathy.interfaces.CHANNEL_TYPE_STREAMED_MEDIA,
81 self.GroupFlagsChanged(0, 0)
82 added, removed = [self._conn.GetSelfHandle()], []
83 localPending, remotePending = [], [contactHandle]
85 '', added, removed, localPending, remotePending,
86 0, telepathy.CHANNEL_GROUP_CHANGE_REASON_NONE
89 def initial_audio(self):
92 def initial_video(self):
95 @misc_utils.log_exception(_moduleLogger)
100 _moduleLogger.debug("Closing call")
101 self._delayedClose.cancel()
103 tp.ChannelTypeStreamedMedia.Close(self)
104 self.remove_from_connection()
106 @misc_utils.log_exception(_moduleLogger)
107 def GetLocalPendingMembersWithInfo(self):
108 info = dbus.Array([], signature="(uuus)")
109 for member in self._local_pending:
110 info.append((member, self._handle, 0, ''))
113 @misc_utils.log_exception(_moduleLogger)
114 def AddMembers(self, handles, message):
115 _moduleLogger.info("Add members %r: %s" % (handles, message))
116 for handle in handles:
117 if handle == int(self.GetSelfHandle()) and self.GetSelfHandle() in self._local_pending:
118 _moduleLogger.info("Technically the user just accepted the call")
120 @misc_utils.log_exception(_moduleLogger)
121 def RemoveMembers(self, handles, message):
122 _moduleLogger.info("Remove members (no-op) %r: %s" % (handles, message))
124 @misc_utils.log_exception(_moduleLogger)
125 def RemoveMembersWithReason(self, handles, message, reason):
126 _moduleLogger.info("Remove members (no-op) %r: %s (%i)" % (handles, message, reason))
128 @misc_utils.log_exception(_moduleLogger)
129 def ListStreams(self):
131 For org.freedesktop.Telepathy.Channel.Type.StreamedMedia
135 @misc_utils.log_exception(_moduleLogger)
136 def RemoveStreams(self, streams):
138 For org.freedesktop.Telepathy.Channel.Type.StreamedMedia
140 raise telepathy.errors.NotImplemented("Cannot remove a stream")
142 @misc_utils.log_exception(_moduleLogger)
143 def RequestStreamDirection(self, stream, streamDirection):
145 For org.freedesktop.Telepathy.Channel.Type.StreamedMedia
147 @note Since streams are short lived, not bothering to implement this
149 _moduleLogger.info("A request was made to change the stream direction")
150 raise telepathy.errors.NotImplemented("Cannot change directions")
152 @misc_utils.log_exception(_moduleLogger)
153 def RequestStreams(self, contactId, streamTypes):
155 For org.freedesktop.Telepathy.Channel.Type.StreamedMedia
157 @returns [(Stream ID, contact, stream type, stream state, stream direction, pending send flags)]
159 contact = self._conn.get_handle_by_id(telepathy.constants.HANDLE_TYPE_CONTACT, contactId)
160 assert self.__contactHandle == contact, "%r != %r" % (self.__contactHandle, contact)
162 le = gobject_utils.AsyncLinearExecution(self._conn.session.pool, self._call)
166 streamState = telepathy.constants.MEDIA_STREAM_STATE_CONNECTED
167 streamDirection = telepathy.constants.MEDIA_STREAM_DIRECTION_BIDIRECTIONAL
168 pendingSendFlags = telepathy.constants.MEDIA_STREAM_PENDING_REMOTE_SEND
169 return [(streamId, contact, streamTypes[0], streamState, streamDirection, pendingSendFlags)]
171 @misc_utils.log_exception(_moduleLogger)
172 def _call(self, contact):
173 contactNumber = contact.phoneNumber
175 self.__calledNumber = contactNumber
176 self.CallStateChanged(self.__contactHandle, telepathy.constants.CHANNEL_CALL_STATE_RINGING)
178 self._delayedClose.start(seconds=0)
179 self.CallStateChanged(self.__contactHandle, telepathy.constants.CHANNEL_CALL_STATE_FORWARDED)
183 self._conn.session.backend.call,
188 _moduleLogger.exception("While placing call to %s" % (self.__calledNumber, ))
189 self._conn.force_log_display()
190 accountNumber = misc_utils.normalize_number(self._conn.session.backend.get_account_number())
191 self._conn.log_to_user(
193 "Error while placing call from %s to %s:\n%s" % (
194 accountNumber, self.__calledNumber, str(e)
199 @misc_utils.log_exception(_moduleLogger)
200 def GetCallStates(self):
202 For org.freedesktop.Telepathy.Channel.Interface.CallState
204 Get the current call states for all contacts involved in this call.
205 @returns {Contact: telepathy.constants.CHANNEL_CALL_STATE_*}
207 return {self.__contactHandle: telepathy.constants.CHANNEL_CALL_STATE_FORWARDED}
209 @misc_utils.log_exception(_moduleLogger)
210 def GetHoldState(self):
212 For org.freedesktop.Telepathy.Channel.Interface.Hold
214 Get the current hold state
215 @returns (HoldState, Reason)
218 telepathy.constants.LOCAL_HOLD_STATE_UNHELD,
219 telepathy.constants.LOCAL_HOLD_STATE_REASON_NONE,
222 @misc_utils.log_exception(_moduleLogger)
223 def RequestHold(self, Hold):
225 For org.freedesktop.Telepathy.Channel.Interface.Hold
229 _moduleLogger.debug("Closing without cancel to get out of users way")
232 @misc_utils.log_exception(_moduleLogger)
233 def _on_close_requested(self, *args):
234 _moduleLogger.debug("Cancel now disallowed")