Newer
Older
pc-autobackup / ssdp.py
#!/usr/bin/python
#
# Copyright 2013 Jeff Rebeiro (jrebeiro@gmail.com) All rights reserved
# Simple SSDP implementation for PC Autobackup

__author__ = 'jrebeiro@gmail.com (Jeff Rebeiro)'

import ConfigParser
import os
import re
import socket
import sys
import uuid
from twisted.internet import reactor
from twisted.internet import task
from twisted.internet.protocol import DatagramProtocol

# TODO(jrebeiro): Move the CONFIG_FILE variable to the main runnable module
CONFIG_FILE = os.path.expanduser("~/pc-autobackup.cfg")

pattern = r'^M-SEARCH.*HOST: (.*):(\d+).*urn:schemas-upnp-org:device:(\w+):1.*'
MSEARCH = re.compile(pattern, re.DOTALL)

SSDP_RESPONSE = ('HTTP/1.1 200 OK\r\n'
                 'CACHE-CONTROL: max-age = 1800\r\n'
                 'EXT:\r\n'
                 'LOCATION: http://%s:52235/DMS/SamsungDmsDesc.xml\r\n'
                 'SERVER: MS-Windows/XP UPnP/1.0 PROTOTYPE/1.0\r\n'
                 'ST: urn:schemas-upnp-org:device:MediaServer:1\r\n'
                 'USN: %s::urn:schemas-upnp-org:device:MediaServer:1\r\n')


class SSDP(DatagramProtocol):

  def __init__(self):
    self.config = LoadOrCreateConfig()
    self.ssdp_address = '239.255.255.250'
    self.ssdp_port = 1900
    self.server = reactor.listenMulticast(self.ssdp_port, self,
                                          listenMultiple=True)
    self.server.setLoopbackMode(1)
    self.server.joinGroup(self.ssdp_address,
                          interface=self.config.get('SSDP',
                                                    'default_interface'))

  def datagramReceived(self, datagram, address):
    m = MSEARCH.match(datagram)
    if m:
      # TODO(jrebeiro): Make this print in debug mode when the main runnable
      #                 module is created and implements optparse
      # TODO(jrebeiro): Verify that MediaServer is the only discovery request
      #                 PCAutoBackup responds to.
      print 'Received M-SEARCH for %s from %r' % (m.group(3), address)
      if m.group(3) == 'MediaServer':
        self.SendSSDPResponse(address)

  def SendSSDPResponse(self, address):
    """Send a response to an SSDP MediaServer discovery request.

    Args:
      address: A tuple of destination IP and Port as strings
    """
    # TODO(jrebeiro): Make this send a UDP response once the HTTP server is
    #                 ready.
    print "Response:"
    print SSDP_RESPONSE % (address[0], self.config.get('SSDP', 'uuid'))

  def stop(self):
    self.server.leaveGroup(self.ssdp_address,
                           interface=self.config.get('SSDP',
                                                     'default_interface'))
    self.server.stopListening()


def LoadOrCreateConfig():
  """Load an existing configuration or create one."""
  config = ConfigParser.RawConfigParser()
  # TODO(jrebeiro): Move the CONFIG_FILE variable to the main runnable module
  config.read(CONFIG_FILE)

  if not config.has_section('SSDP'):
    config.add_section('SSDP')
    config.set('SSDP', 'uuid', uuid.uuid4())
    config.set('SSDP', 'default_interface',
               socket.gethostbyname(socket.gethostname()))
    with open(CONFIG_FILE, 'wb') as config_file:
      config.write(config_file)

  return config


def SSDPReactor():
  """Callback function for twisted.internet.reactor."""
  ssdp_server = SSDP()
  reactor.addSystemEventTrigger('before', 'shutdown', ssdp_server.stop)


def main():
  # TODO(jrebeiro): Move this code to a main runnable module
  config = LoadOrCreateConfig()
  reactor.callWhenRunning(SSDPReactor)
  reactor.run()


if __name__ == "__main__":
  main()