cybertools/twisted/manhole.py
helmutm af872764f5 manhole can now be opend/closed TTW
git-svn-id: svn://svn.cy55.de/Zope3/src/cybertools/trunk@1102 fd906abe-77d9-0310-91a1-e0d9ade77398
2006-03-01 10:47:55 +00:00

157 lines
5.4 KiB
Python

"""
A simple twisted manhole that allows you to access a running Zope 3
instance via a python command line without having to run ZEO.
You may run it for testing purposes via `python manhole.py` (note that
the twisted library must be reachable via your PYTHONPATH) and log in
from another console window using `ssh -p 5001 admin@localhost`. The
password is defined below in the "reactor.listenTCP()" statement. The
manhole script may be stopped with Ctrl-C.
Note that this will open up a serious security hole on your computer
as now anybody knowing this password may login to the Python console
and get full access to the system with the permissions of the user
running the manhole script.
In order to use it with Zope copy the cybertools.twisted-configure.zcml
to the etc/package-includes directory of your Zope instance and restart
Zope. Open the manhole via the "Manhole Control" Tab of the "Manage process"
menu.
You can then log in with ssh like shown above, using the username
and password of the zope.manager principal defined in your principals.zcml.
After logging in use the `help` command to get more information.
$Id$
"""
from twisted.internet import reactor, protocol, defer
from twisted.protocols import basic
from twisted.cred import portal, checkers, credentials, error as credError
from twisted.conch import manhole as manhole, manhole_ssh
from zope.interface import implements
try:
from zope.app.publication.zopepublication import ZopePublication
from zope.app.component.hooks import setSite
from zope.app.security.principalregistry import principalRegistry
from zope.app.security import settings
from zope.app.security.interfaces import IAuthentication
from zope.app.securitypolicy.principalrole import principalRoleManager
from zope.app import zapi
import transaction
hasZope = True
except:
hasZope = False
import time
import sys
from cStringIO import StringIO
listener = None
factory = None
printLog = None
port = 5001
def getManholeFactory(namespace, **passwords):
realm = manhole_ssh.TerminalRealm()
def getManhole(_):
#return manhole.ColoredManhole(namespace)
return manhole.Manhole(namespace)
realm.chainedProtocolFactory.protocolFactory = getManhole
p = portal.Portal(realm)
checker = (hasZope and ZopeManagerChecker() or
checkers.InMemoryUsernamePasswordDatabaseDontUse(**passwords))
p.registerChecker(checker)
return manhole_ssh.ConchFactory(p)
class ZopeManagerChecker(object):
implements(checkers.ICredentialsChecker)
credentialInterfaces = (credentials.IUsernamePassword,)
def requestAvatarId(self, credentials):
login = credentials.username
password = credentials.password
# TODO: This should be based on the official Zope API stuff, e.g. via:
#principalRegistry = zapi.getUtility(IAuthentication)
principal = principalRegistry.getPrincipalByLogin(login)
if principal.validate(password):
roles = principalRoleManager.getRolesForPrincipal(principal.id)
for role, setting in roles:
if role == 'zope.Manager' and setting == settings.Allow:
return defer.succeed(login)
return defer.fail(credError.UnauthorizedLogin(
'Insufficient permissions'))
return defer.fail(credError.UnauthorizedLogin(
'User/password not correct'))
def printTime():
global printLog
print '***', time.strftime('%H:%M:%S'), '- twisted.manhole open ***'
printLog = reactor.callLater(600, printTime)
class Help(object):
def __repr__(self):
info = """
Use `dir()` to see what variables and functions are available.
"""
zopeInfo = """
You may use `x = zapi.traverse(root, 'path/to/object')` to get an
object in your folder hierarchy. Then you may call any method or
access any attribute of this object.
In order to get access to local utilities and adapters you may
issue a `setSite(root)`. Don't forget to call `setSite()` before
finishing your session in order to reset this setting.
If you change an object stored in the ZODB you should issue a
`transaction.commit()` to make your changes permanent or a
`transaction.abort()` to undo them.
"""
return info + (hasZope and zopeInfo or '')
def __call__(self):
print self
help = Help()
def open(port=5001, request=None):
global hasZope, factory, listener
printTime()
d = globals()
if hasZope and request is not None:
database = request.publication.db
connection = database.open()
root = connection.root()[ZopePublication.root_name]
else:
hasZope = False
d.update(locals())
namespace = {}
for key in ('__builtins__', 'connection', 'event', 'setSite', 'hasZope',
'zapi', 'transaction', 'root', '__doc__', 'help',
'manholeFactory', 'context'):
if key in d:
namespace[key] = d[key]
# TODO: get admin password from somewhere else or use a real checker.
factory = getManholeFactory(namespace, admin='aaa')
listener = reactor.listenTCP(port, factory)
def close():
global listener
listener.stopListening()
listener = None
if __name__ == '__main__':
port = 5001
if len(sys.argv) > 1:
port = int(sys.argv[-1])
startup(port=port)
reactor.run()