6 from threading
import Timer
7 from urllib
import quote
12 from plugin
import GetPlugin
14 SHARE_TEMPLATE
= '/TiVoConnect?Command=QueryContainer&Container=%s'
15 PLATFORM_MAIN
= 'pyTivo'
16 PLATFORM_VIDEO
= 'pc/pyTivo' # For the nice icon
19 def __init__(self
, names
):
22 def removeService(self
, server
, type, name
):
23 if name
in self
.names
:
24 self
.names
.remove(name
)
26 def addService(self
, server
, type, name
):
27 self
.names
.append(name
)
30 def __init__(self
, logger
):
31 """ Announce our shares via Zeroconf. """
35 self
.rz
= Zeroconf
.Zeroconf()
36 address
= inet_aton(config
.get_ip())
37 port
= int(config
.getPort())
38 for section
, settings
in config
.getShares():
39 ct
= GetPlugin(settings
['type']).CONTENT_TYPE
40 if ct
.startswith('x-container/'):
42 platform
= PLATFORM_VIDEO
44 platform
= PLATFORM_MAIN
45 logger
.info('Registering: %s' % section
)
46 self
.share_names
.append(section
)
47 desc
= {'path': SHARE_TEMPLATE
% quote(section
),
48 'platform': platform
, 'protocol': 'http'}
50 info
= Zeroconf
.ServiceInfo('_%s._tcp.local.' % tt
,
51 '%s._%s._tcp.local.' % (section
, tt
),
52 address
, port
, 0, 0, desc
)
53 self
.rz
.registerService(info
)
54 self
.share_info
.append(info
)
57 """ Look for TiVos using Zeroconf. """
58 VIDS
= '_tivo-videos._tcp.local.'
61 # Get the names of servers offering TiVo videos
62 browser
= Zeroconf
.ServiceBrowser(self
.rz
, VIDS
, ZCListener(names
))
64 # Give them half a second to respond
67 # Now get the addresses -- this is the slow part
69 info
= self
.rz
.getServiceInfo(VIDS
, name
)
70 if info
and 'TSN' in info
.properties
:
71 tsn
= info
.properties
['TSN']
72 address
= inet_ntoa(info
.getAddress())
73 config
.tivos
[tsn
] = address
74 name
= name
.replace('.' + VIDS
, '')
75 self
.logger
.info(name
)
76 config
.tivo_names
[tsn
] = name
79 self
.logger
.info('Unregistering: %s' % ' '.join(self
.share_names
))
80 for info
in self
.share_info
:
81 self
.rz
.unregisterService(info
)
86 self
.UDPSock
= socket(AF_INET
, SOCK_DGRAM
)
87 self
.UDPSock
.setsockopt(SOL_SOCKET
, SO_BROADCAST
, 1)
90 self
.platform
= PLATFORM_VIDEO
91 for section
, settings
in config
.getShares():
92 ct
= GetPlugin(settings
['type']).CONTENT_TYPE
93 if ct
in ('x-container/tivo-music', 'x-container/tivo-photos'):
94 self
.platform
= PLATFORM_MAIN
98 logger
= logging
.getLogger('pyTivo.beacon')
100 logger
.info('Announcing shares...')
101 self
.bd
= ZCBroadcast(logger
)
103 logger
.error('Zeroconf failure')
106 logger
.info('Scanning for TiVos...')
111 def add_service(self
, service
):
112 self
.services
.append(service
)
115 def format_services(self
):
116 return ';'.join(self
.services
)
118 def format_beacon(self
, conntype
, services
=True):
119 beacon
= ['tivoconnect=1',
120 'method=%s' % conntype
,
121 'identity=%s' % config
.getGUID(),
122 'machine=%s' % gethostname(),
123 'platform=%s' % self
.platform
]
126 beacon
.append('services=' + self
.format_services())
128 beacon
.append('services=TiVoMediaServer:0/http')
130 return '\n'.join(beacon
)
132 def send_beacon(self
):
133 beacon_ips
= config
.getBeaconAddresses()
134 for beacon_ip
in beacon_ips
.split():
135 if beacon_ip
!= 'listen':
137 self
.UDPSock
.sendto(self
.format_beacon('broadcast'),
144 self
.timer
= Timer(60, self
.start
)
153 """ For the direct-connect, TCP-style beacon """
157 TCPSock
= socket(AF_INET
, SOCK_STREAM
)
158 TCPSock
.bind(('', 2190))
162 # Wait for a connection
163 client
, address
= TCPSock
.accept()
165 # Accept the client's beacon
166 client_length
= struct
.unpack('!I', client
.recv(4))[0]
167 client_message
= client
.recv(client_length
)
170 message
= self
.format_beacon('connected')
171 client
.send(struct
.pack('!I', len(message
)))
175 thread
.start_new_thread(server
, ())
177 def get_name(self
, address
):
178 """ Exchange beacons, and extract the machine name. """
179 our_beacon
= self
.format_beacon('connected', False)
180 machine_name
= re
.compile('machine=(.*)\n').search
184 tsock
.connect((address
, 2190))
186 tsock
.send(struct
.pack('!I', len(our_beacon
)))
187 tsock
.send(our_beacon
)
189 length
= struct
.unpack('!I', tsock
.recv(4))[0]
190 tivo_beacon
= tsock
.recv(length
)
194 name
= machine_name(tivo_beacon
).groups()[0]