Как спавнить браузер

Я разрабатываю irc-клиент на python на основе irc.IRCClient и pygtk, я использую правильный реактор, и все работает нормально. Теперь я бы запускал браузер при нажатии URL-адреса... Лучше использовать xdg-open, который запускает настроенный браузер по умолчанию (в бесплатном рабочем столе, совместимом с DE). URL-адрес выбирается в событии нажатия кнопки gtk. Я перепробовал все возможные варианты, но когда-либо процессор загружался на 100%. Ниже приведены различные способы, которые я пробовал, только использование реактора.spawnProcess кажется многообещающим, потому что, пока браузер не открыт, все в порядке, при его закрытии процессор достигает 100%. Все остальные процессоры скоро скачут до 100% и останутся. Запуск браузера напрямую ничего не меняет. Что я делаю неправильно?

питон 2.6.5 - скрученный 10.1.0 - pygtk 2.16.0

Marco

def on_click(self, b):
    .....
    .....
    if data:
        url = self.urls[int(data)]

        # 100% cpu forever
        browser = utils.getProcessValue('/usr/bin/xdg-open', [url,], os.environ)
        browser.addCallback(self.printExitValue)

        # 100% cpu after closing browser 
        xdgProcess = XDGProcessProtocol()
        reactor.spawnProcess(xdgProcess, '/usr/bin/xdg-open', ['/usr/bin/xdg-open', url], os.environ )    

        # 100% cpu forever
        os.system('xdg-open %s' % url)

        # 100% cpu forever
        os.spawnl(os.P_NOWAIT, '/usr/bin/xdg-open', '/usr/bin/xdg-open', url)

        # 100% cpu forever    
        reactor.callInThread(self.browser, url)

        return 0

def printExitValue(self, val):
    print 'xdg-open %d' % val

def browser(self, url):
    os.spawnl(os.P_NOWAIT, '/usr/bin/xdg-open', '/usr/bin/xdg-open', url)

class XDGProcessProtocol(protocol.ProcessProtocol):

def __init__(self):
    self.data = ''

def connectionMade(self):
    pass

def outReceived(self, data):
    self.data = self.data + data

def errReceived(self, data):
    self.data = self.data + data

def inConnectionLost(self):
    pass

def outConnectionLost(self):
    print self.data

def errConnectionLost(self):
    pass

def processExited(self, reason):
    print "processExited, status %d" % (reason.value.exitCode,)

def processEnded(self, reason):
    print "processEnded, status %d" % (reason.value.exitCode,)
    print "quitting"


person Marco    schedule 01.09.2010    source источник
comment
Я понятия не имею, о чем вы спрашиваете в вопросе. Однако, чтобы открыть веб-браузер, просто: import webbrowser; webbrowser.open(url)   -  person Blue Peppers    schedule 01.09.2010
comment
Было бы полезно, если бы вы могли предоставить простую программу, которая раскрывает вашу проблему, но при этом как можно меньше (если возможно, без IRC-вещей и т. д.).   -  person JanC    schedule 01.09.2010
comment
Вы пытались использовать отладчик, такой как WinPDB, чтобы выяснить, что он на самом деле делает, чтобы вызвать 100% загрузку ЦП? Вот с чего я бы начал.   -  person ssokolow    schedule 05.09.2010


Ответы (4)


Чтобы закрыть тему: это ошибка как в pygobject, так и в pygtk, обработчик SIGCHLD входит в бесконечный цикл. Это было исправлено в pygobject-2.21,0 и pygtk-2.17.0.

Марко

person Marco    schedule 07.09.2010
comment
Не стесняйтесь принимать свой собственный ответ. Тем не менее, ссылки на отчеты об ошибках были бы полезны... - person Michael Wild; 15.04.2013

Вы можете использовать это:

import webbrowser

webbrowser.open("http://www.google.it/")
person necrox87    schedule 15.04.2013

Модуль веб-браузера, возможно, то, что вы ищете? Это часть стандартной библиотеки Python.

person Ivo van der Wijk    schedule 01.09.2010
comment
Я хочу запустить браузер, такой как firefox, konqueror и т. д., в среде рабочего стола, такой как Gnome или KDE, у вас настроен браузер по умолчанию, который запускается при нажатии URL-адреса, например. в окне терминала. xdg-open — это утилита, используемая для этого. - person Marco; 01.09.2010
comment
Я пытался использовать веб-браузер, он работает, запуская предопределенный браузер, но чашка все равно достигает 100% - person Marco; 01.09.2010

Вот рабочий (и неудачный) пример.

Я адаптировал ваш irclogbot.py для запуска в приложении pygtk.

Sys.argv[1] — это URL-адрес irc-сервера.

В методе on_click измените путь к исполняемому файлу в соответствии с вашим браузером.

После того, как браузер открыл страницу, закройте ее, загрузка процессора подскочит до 100%.

#!/usr/bin/env python
# -*- coding: utf-8 -*-

# Copyright (c) 2001-2009 Twisted Matrix Laboratories.
# See LICENSE for details.
import gobject
import pygtk
import gtk

# twisted imports
from twisted.internet import gtk2reactor
gtk2reactor.install()
from twisted.words.protocols import irc
from twisted.internet import reactor, protocol
from twisted.python import log

# system imports
import time, sys

class MessageLogger:
    """
    An independent logger class (because separation of application
    and protocol logic is a good thing).
    """
    def __init__(self, file):
        self.file = file

    def log(self, message):
        """Write a message to the file."""
        timestamp = time.strftime("[%H:%M:%S]", time.localtime(time.time()))
        self.file.write('%s %s\n' % (timestamp, message))
        self.file.flush()

    def close(self):
        self.file.close()


class LogBot(irc.IRCClient):
    """A logging IRC bot."""

    nickname = "twistedbot"

    def connectionMade(self):
        irc.IRCClient.connectionMade(self)
        self.logger = MessageLogger(open(self.factory.filename, "a"))
        self.logger.log("[connected at %s]" % 
                        time.asctime(time.localtime(time.time())))

    def connectionLost(self, reason):
        irc.IRCClient.connectionLost(self, reason)
        self.logger.log("[disconnected at %s]" % 
                        time.asctime(time.localtime(time.time())))
        self.logger.close()


    # callbacks for events

    def signedOn(self):
        """Called when bot has succesfully signed on to server."""
        self.join(self.factory.channel)

    def joined(self, channel):
        """This will get called when the bot joins the channel."""
        self.logger.log("[I have joined %s]" % channel)

    def privmsg(self, user, channel, msg):
        """This will get called when the bot receives a message."""
        user = user.split('!', 1)[0]
        self.logger.log("<%s> %s" % (user, msg))

        # Check to see if they're sending me a private message
        if channel == self.nickname:
            msg = "It isn't nice to whisper!  Play nice with the group."
            self.msg(user, msg)
            return

        # Otherwise check to see if it is a message directed at me
        if msg.startswith(self.nickname + ":"):
            msg = "%s: I am a log bot" % user
            self.msg(channel, msg)
            self.logger.log("<%s> %s" % (self.nickname, msg))

    def action(self, user, channel, msg):
        """This will get called when the bot sees someone do an action."""
        user = user.split('!', 1)[0]
        self.logger.log("* %s %s" % (user, msg))

    # irc callbacks

    def irc_NICK(self, prefix, params):
        """Called when an IRC user changes their nickname."""
        old_nick = prefix.split('!')[0]
        new_nick = params[0]
        self.logger.log("%s is now known as %s" % (old_nick, new_nick))


    # For fun, override the method that determines how a nickname is changed on
    # collisions. The default method appends an underscore.
    def alterCollidedNick(self, nickname):
        """
        Generate an altered version of a nickname that caused a collision in an
        effort to create an unused related name for subsequent registration.
        """
        return nickname + '^'



class LogBotFactory(protocol.ClientFactory):
    """A factory for LogBots.

    A new protocol instance will be created each time we connect to the server.
    """

    # the class of the protocol to build when new connection is made
    protocol = LogBot

    def __init__(self, channel, filename):
        self.channel = channel
        self.filename = filename

    def clientConnectionLost(self, connector, reason):
        """If we get disconnected, reconnect to server."""
        connector.connect()

    def clientConnectionFailed(self, connector, reason):
        print "connection failed:", reason
        reactor.stop()

class GUI(object):
    def __init__(self):
        self.mw = gtk.Window()
        self.mw.connect('destroy', self.quit)
        bt = gtk.Button('Run browser')
        bt.connect('clicked', self.on_click)
        frame = gtk.Frame('Click me')
        frame.add(bt)
        self.mw.add(frame)
        self.mw.show_all()
        f = LogBotFactory('#prova', 'botlog.txt')
        # connect factory to this host and port
        reactor.connectTCP(sys.argv[1], 6667, f)
        reactor.run()

    def on_click(self, b):
        url = 'http://www.gentoo.org'
        xdgProcess = XDGProcessProtocol()
        #####################################################
        # change the executable path of the browser you have
        #####################################################
        reactor.spawnProcess(xdgProcess, '/usr/bin/midori', ['/usr/bin/midori', url], None)    
        print 'clicked'

    def quit(self, w):
        print 'closeapp'
        try:
            reactor.stop()
        except:
            pass
        gtk.main_quit()

class XDGProcessProtocol(protocol.ProcessProtocol):
    def __init__(self):
        self.data = ''

    def connectionMade(self):
        pass

    def outReceived(self, data):
        self.data = self.data + data

    def errReceived(self, data):
        self.data = self.data + data
    def inConnectionLost(self):
        pass

    def outConnectionLost(self):
        print "outConnectionLost! The child closed their stdout!"
        print self.data

    def errConnectionLost(self):
        pass

    def processExited(self, reason):
        print "processExited, status %d" % (reason.value.exitCode,)

    def processEnded(self, reason):
        print "processEnded, status %d" % (reason.value.exitCode,)
        print "quitting"

if __name__ == '__main__':
    #########################################    
    # sys.argv[1] is the url of the irc server    
    #########################################
    # initialize logging
    log.startLogging(sys.stdout)

    GUI()
    gtk.main()

Марко

person Marco    schedule 02.09.2010