A Python UPnP Media Server

142 lines
3.7 KiB

  1. #!/usr/bin/env python
  2. # Licensed under the MIT license
  3. # http://opensource.org/licenses/mit-license.php
  4. # Copyright 2005, Tim Potter <tpot@samba.org>
  5. from DIDLLite import TextItem, AudioItem, VideoItem, Resource
  6. import os
  7. import os.path
  8. import random
  9. import socket
  10. import string
  11. import sys
  12. import traceback
  13. from twisted.python import log
  14. from twisted.internet import reactor
  15. def generateuuid():
  16. if False:
  17. return 'uuid:asdflkjewoifjslkdfj'
  18. return ''.join([ 'uuid:'] + map(lambda x: random.choice(string.letters), xrange(20)))
  19. listenAddr = sys.argv[1]
  20. if len(sys.argv) > 2:
  21. listenPort = int(sys.argv[2])
  22. if listenPort < 1024 or listenPort > 65535:
  23. raise ValueError, 'port out of range'
  24. else:
  25. listenPort = random.randint(10000, 65000)
  26. log.startLogging(sys.stdout)
  27. # Create SSDP server
  28. from SSDP import SSDPServer, SSDP_PORT, SSDP_ADDR
  29. s = SSDPServer()
  30. port = reactor.listenMulticast(SSDP_PORT, s)
  31. port.joinGroup(SSDP_ADDR)
  32. port.setLoopbackMode(0) # don't get our own sends
  33. uuid = generateuuid()
  34. urlbase = 'http://%s:%d/' % (listenAddr, listenPort)
  35. # Create SOAP server
  36. from twisted.web import server, resource, static
  37. from ContentDirectory import ContentDirectoryServer
  38. from ConnectionManager import ConnectionManagerServer
  39. class WebServer(resource.Resource):
  40. def __init__(self):
  41. resource.Resource.__init__(self)
  42. class RootDevice(static.Data):
  43. def __init__(self):
  44. r = {
  45. 'hostname': socket.gethostname(),
  46. 'uuid': uuid,
  47. 'urlbase': urlbase,
  48. }
  49. d = file('root-device.xml').read() % r
  50. static.Data.__init__(self, d, 'text/xml')
  51. root = WebServer()
  52. cds = ContentDirectoryServer('My Media Server')
  53. root.putChild('ContentDirectory', cds)
  54. cds = cds.control
  55. root.putChild('ConnectionManager', ConnectionManagerServer())
  56. root.putChild('root-device.xml', RootDevice())
  57. # Area of server to serve media files from
  58. from MediaServer import MediaServer
  59. medianode = static.File('media')
  60. medianode.contentTypes.update( {
  61. '.wmv': 'video/x-ms-wmv',
  62. '.ts': 'video/mp2t',
  63. #'.ts': 'video/mpeg', # we may want this instead of mp2t
  64. '.mp4': 'video/mp4',
  65. })
  66. root.putChild('media', medianode)
  67. # Set up media files
  68. allmedia = cds.addContainer('0', 'All Media')
  69. log.msg('allmedia: %s' % repr(allmedia))
  70. for i in os.listdir('media'):
  71. fpath = os.path.join('media', i)
  72. try:
  73. if not os.path.isfile(fpath):
  74. log.msg('skipping: %s' % fpath)
  75. continue
  76. fn, ext = os.path.splitext(i)
  77. mt = medianode.contentTypes[ext]
  78. ty = mt.split('/')[0]
  79. if ty == 'video':
  80. klass = VideoItem
  81. elif ty == 'audio':
  82. klass = AudioItem
  83. elif ty == 'text':
  84. klass = TextItem
  85. else:
  86. raise KeyError, 'no item for mt: %s' % mt
  87. item = cds.addItem(allmedia, klass, fn)
  88. cds[item].res = Resource('%smedia/%s' % (urlbase, i), 'http-get:*:%s:*' % mt)
  89. cds[item].res.size = os.path.getsize(fpath)
  90. except KeyError:
  91. traceback.print_exc(file=log.logfile)
  92. site = server.Site(root)
  93. reactor.listenTCP(listenPort, site)
  94. # we need to do this after the children are there, since we send notifies
  95. s.register('%s::upnp:rootdevice' % uuid,
  96. 'upnp:rootdevice',
  97. urlbase + 'root-device.xml')
  98. s.register(uuid,
  99. uuid,
  100. urlbase + 'root-device.xml')
  101. s.register('%s::urn:schemas-upnp-org:device:MediaServer:1' % uuid,
  102. 'urn:schemas-upnp-org:device:MediaServer:1',
  103. urlbase + 'root-device.xml')
  104. s.register('%s::urn:schemas-upnp-org:service:ConnectionManager:1' % uuid,
  105. 'urn:schemas-upnp-org:device:ConnectionManager:1',
  106. urlbase + 'root-device.xml')
  107. s.register('%s::urn:schemas-upnp-org:service:ContentDirectory:1' % uuid,
  108. 'urn:schemas-upnp-org:device:ContentDirectory:1',
  109. urlbase + 'root-device.xml')
  110. # Main loop
  111. reactor.run()