使用Python将消息推送到客户端

时间:2011-07-27 16:59:10

标签: python real-time

如何在必要时实施能够将消息“推送”到客户端的系统? 基本上,我需要的是能够从Python服务器“推送”文本到Python客户端。然后将文本解析为将在客户端处理的操作(一旦收到'action-text',我已经知道如何做这件事)。 我可以每隔几秒检查一下正在等待的新“动作文本” - 我认为这对于成千上万的客户来说并不可靠且可扩展。实时的事情在这里非常重要。

有什么建议吗?

2 个答案:

答案 0 :(得分:7)

您可以使用redis发布订阅模型more here

Redis具有高度可扩展性和快速性。

示例:(来自https://github.com/andymccurdy/redis-py/blob/master/tests/pubsub.py

import redis
import unittest

class PubSubTestCase(unittest.TestCase):
    def setUp(self):
        self.connection_pool = redis.ConnectionPool()
        self.client = redis.Redis(connection_pool=self.connection_pool)
        self.pubsub = self.client.pubsub()

    def tearDown(self):
        self.connection_pool.disconnect()

    def test_channel_subscribe(self):
        self.assertEquals(
            self.pubsub.subscribe('foo'),
            ['subscribe', 'foo', 1]
            )
        self.assertEquals(self.client.publish('foo', 'hello foo'), 1)
        self.assertEquals(
            self.pubsub.listen().next(),
            {
                'type': 'message',
                'pattern': None,
                'channel': 'foo',
                'data': 'hello foo'
            }
            )
        self.assertEquals(
            self.pubsub.unsubscribe('foo'),
            ['unsubscribe', 'foo', 0]
            )

    def test_pattern_subscribe(self):
        self.assertEquals(
            self.pubsub.psubscribe('fo*'),
            ['psubscribe', 'fo*', 1]
            )
        self.assertEquals(self.client.publish('foo', 'hello foo'), 1)
        self.assertEquals(
            self.pubsub.listen().next(),
            {
                'type': 'pmessage',
                'pattern': 'fo*',
                'channel': 'foo',
                'data': 'hello foo'
            }
            )
        self.assertEquals(
            self.pubsub.punsubscribe('fo*'),
            ['punsubscribe', 'fo*', 0]
            )

答案 1 :(得分:2)

我建议让每个客户端建立到服务器的套接字连接。每个客户端都可以运行一个线程,该线程尝试从套接字进行阻塞读取(将其置于后台线程中),该线程在套接字可用时从套接字中提取信息。然后,每当服务器写入该套接字时,客户端应立即获取它。客户端的连接应该是telnetlib.telnet imho的子类,因为它提供了方便的read_until方法。这样的事情可能是:

class RPCConnection(object, Telnet):
  def __init__(self, host = 'localhost', port = 9198, auto = True):
    Telnet.__init__(self)

    self.host, self.port, self.connected = host, port, False

    if auto:
      self._connect()

  def _connect(self):
    self.open(self.host, self.port)

    self.connected = True

  def _disconnect(self):
    self.close()

    self.connected = False

  def _send(self, dict):
    self.write(json.dumps(dict))

  def _recv(self):
    resp = self.read_until('\n')

    try:
      return json.loads(resp)
    except Exception, e:
      print e
      print resp