[project @ Arch-1:robey@lag.net--2003-public%secsh--dev--1.0--patch-56]
add forward.py demo script; bump to gyarados add a demo script to show how to do local port forwarding. add gyarados to all the docs and bump the version number everywhere.
This commit is contained in:
parent
4d30633457
commit
9baa2b361e
|
@ -1,3 +1,3 @@
|
||||||
include ChangeLog LICENSE test.py demo.py demo_simple.py demo_server.py demo_dss_key demo_rsa_key
|
include ChangeLog LICENSE test.py demo.py demo_simple.py demo_server.py demo_dss_key demo_rsa_key forward.py
|
||||||
recursive-include docs *
|
recursive-include docs *
|
||||||
recursive-include tests *.py
|
recursive-include tests *.py
|
||||||
|
|
1
Makefile
1
Makefile
|
@ -5,6 +5,7 @@
|
||||||
# doduo (04jan04) - 0.9
|
# doduo (04jan04) - 0.9
|
||||||
# eevee (08mar04)
|
# eevee (08mar04)
|
||||||
# fearow (23apr04)
|
# fearow (23apr04)
|
||||||
|
# gyarados (31may04)
|
||||||
|
|
||||||
release:
|
release:
|
||||||
python ./setup.py sdist --formats=zip
|
python ./setup.py sdist --formats=zip
|
||||||
|
|
11
README
11
README
|
@ -1,5 +1,5 @@
|
||||||
paramiko 0.9
|
paramiko 0.9
|
||||||
"fearow" release, 23 apr 2004
|
"gyarados" release, 31 may 2004
|
||||||
|
|
||||||
Copyright (c) 2003-2004 Robey Pointer <robey@lag.net>
|
Copyright (c) 2003-2004 Robey Pointer <robey@lag.net>
|
||||||
|
|
||||||
|
@ -117,6 +117,15 @@ the best and easiest examples of how to use the SFTP class.
|
||||||
|
|
||||||
highlights of what's new in each release:
|
highlights of what's new in each release:
|
||||||
|
|
||||||
|
v0.9 GYARADOS
|
||||||
|
* Transport.open_channel() -- supports local & remote port forwarding now
|
||||||
|
* now imports UTF-8 encodings explicitly as a hint to "freeze" utilities
|
||||||
|
* no longer rejects older SFTP servers
|
||||||
|
* default packet size bumped to 8kB
|
||||||
|
* fixed deadlock in closing a channel
|
||||||
|
* Transport.connect() -- fixed bug where it would always fail when given a
|
||||||
|
host key to verify
|
||||||
|
|
||||||
v0.9 FEAROW
|
v0.9 FEAROW
|
||||||
* Transport.send_ignore() -- send random ignored bytes
|
* Transport.send_ignore() -- send random ignored bytes
|
||||||
* RSAKey/DSSKey added from_private_key_file() as a factory constructor;
|
* RSAKey/DSSKey added from_private_key_file() as a factory constructor;
|
||||||
|
|
|
@ -0,0 +1,210 @@
|
||||||
|
#!/usr/bin/python
|
||||||
|
|
||||||
|
"""
|
||||||
|
Sample script showing how to do local port forwarding over paramiko.
|
||||||
|
|
||||||
|
This script connects to the requested SSH server and sets up local port
|
||||||
|
forwarding (the openssh -L option) from a local port through a tunneled
|
||||||
|
connection to a destination reachable from the SSH server machine.
|
||||||
|
|
||||||
|
It uses SocketServer and select, so may not work on Windows.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
import socket
|
||||||
|
import select
|
||||||
|
import SocketServer
|
||||||
|
import getpass
|
||||||
|
import base64
|
||||||
|
from optparse import OptionParser
|
||||||
|
|
||||||
|
import paramiko
|
||||||
|
|
||||||
|
DEFAULT_PORT = 4000
|
||||||
|
SSH_PORT = 22
|
||||||
|
VERBOSE = True
|
||||||
|
READPASS = False
|
||||||
|
|
||||||
|
|
||||||
|
class ForwardServer (SocketServer.ThreadingTCPServer):
|
||||||
|
daemon_threads = True
|
||||||
|
allow_reuse_address = True
|
||||||
|
|
||||||
|
|
||||||
|
class Handler (SocketServer.BaseRequestHandler):
|
||||||
|
|
||||||
|
def handle(self):
|
||||||
|
try:
|
||||||
|
chan = self.ssh_transport.open_channel('direct-tcpip',
|
||||||
|
(self.chain_host, self.chain_port),
|
||||||
|
self.request.getpeername())
|
||||||
|
except Exception, e:
|
||||||
|
verbose('Incoming request to %s:%d failed: %s' % (self.chain_host,
|
||||||
|
self.chain_port,
|
||||||
|
repr(e)))
|
||||||
|
return
|
||||||
|
|
||||||
|
verbose('Connected! Tunnel open.')
|
||||||
|
while True:
|
||||||
|
r, w, x = select.select([self.request, chan], [], [])
|
||||||
|
if self.request in r:
|
||||||
|
data = self.request.recv(1024)
|
||||||
|
if len(data) == 0:
|
||||||
|
break
|
||||||
|
chan.send(data)
|
||||||
|
if chan in r:
|
||||||
|
data = chan.recv(1024)
|
||||||
|
if len(data) == 0:
|
||||||
|
break
|
||||||
|
self.request.send(data)
|
||||||
|
chan.close()
|
||||||
|
self.request.close()
|
||||||
|
verbose('Tunnel closed.')
|
||||||
|
|
||||||
|
|
||||||
|
def forward_tunnel(local_port, remote_host, remote_port, transport):
|
||||||
|
# this is a little convoluted, but lets me configure things for the Handler
|
||||||
|
# object. (SocketServer doesn't give Handlers any way to access the outer
|
||||||
|
# server normally.)
|
||||||
|
class SubHander (Handler):
|
||||||
|
chain_host = remote_host
|
||||||
|
chain_port = remote_port
|
||||||
|
ssh_transport = transport
|
||||||
|
ForwardServer(('', local_port), SubHander).serve_forever()
|
||||||
|
|
||||||
|
def load_host_keys():
|
||||||
|
filename = os.path.expanduser('~/.ssh/known_hosts')
|
||||||
|
keys = {}
|
||||||
|
try:
|
||||||
|
f = open(filename, 'r')
|
||||||
|
except Exception, e:
|
||||||
|
print '*** Unable to open host keys file (%s)' % filename
|
||||||
|
return
|
||||||
|
for line in f:
|
||||||
|
keylist = line.split(' ')
|
||||||
|
if len(keylist) != 3:
|
||||||
|
continue
|
||||||
|
hostlist, keytype, key = keylist
|
||||||
|
hosts = hostlist.split(',')
|
||||||
|
for host in hosts:
|
||||||
|
if not keys.has_key(host):
|
||||||
|
keys[host] = {}
|
||||||
|
keys[host][keytype] = base64.decodestring(key)
|
||||||
|
f.close()
|
||||||
|
return keys
|
||||||
|
|
||||||
|
def find_default_key_file():
|
||||||
|
filename = os.path.expanduser('~/.ssh/id_rsa')
|
||||||
|
if os.access(filename, os.R_OK):
|
||||||
|
return filename
|
||||||
|
filename = os.path.expanduser('~/.ssh/id_dsa')
|
||||||
|
if os.access(filename, os.R_OK):
|
||||||
|
return filename
|
||||||
|
return ''
|
||||||
|
|
||||||
|
def verbose(s):
|
||||||
|
if VERBOSE:
|
||||||
|
print s
|
||||||
|
|
||||||
|
|
||||||
|
#####
|
||||||
|
|
||||||
|
|
||||||
|
parser = OptionParser(usage='usage: %prog [options] <remote-addr>:<remote-port>',
|
||||||
|
version='%prog 1.0')
|
||||||
|
parser.add_option('-q', '--quiet', action='store_false', dest='verbose', default=VERBOSE,
|
||||||
|
help='squelch all informational output')
|
||||||
|
parser.add_option('-l', '--local-port', action='store', type='int', dest='port',
|
||||||
|
default=DEFAULT_PORT,
|
||||||
|
help='local port to forward (default: %d)' % DEFAULT_PORT)
|
||||||
|
parser.add_option('-r', '--host', action='store', type='string', dest='ssh_host',
|
||||||
|
help='SSH host to tunnel through (required)')
|
||||||
|
parser.add_option('-p', '--port', action='store', type='int', dest='ssh_port', default=SSH_PORT,
|
||||||
|
help='SSH port to tunnel through (default: %d)' % SSH_PORT)
|
||||||
|
parser.add_option('-u', '--user', action='store', type='string', dest='user',
|
||||||
|
default=getpass.getuser(),
|
||||||
|
help='username for SSH authentication (default: %s)' % getpass.getuser())
|
||||||
|
parser.add_option('-K', '--key', action='store', type='string', dest='keyfile',
|
||||||
|
default=find_default_key_file(),
|
||||||
|
help='private key file to use for SSH authentication')
|
||||||
|
parser.add_option('', '--no-key', action='store_false', dest='use_key', default=True,
|
||||||
|
help='don\'t look for or use a private key file')
|
||||||
|
parser.add_option('-P', '--password', action='store_true', dest='readpass', default=READPASS,
|
||||||
|
help='read password (for key or password auth) from stdin')
|
||||||
|
options, args = parser.parse_args()
|
||||||
|
|
||||||
|
VERBOSE = options.verbose
|
||||||
|
READPASS = options.readpass
|
||||||
|
|
||||||
|
|
||||||
|
if len(args) != 1:
|
||||||
|
parser.error('Incorrect number of arguments.')
|
||||||
|
remote_host = args[0]
|
||||||
|
if ':' not in remote_host:
|
||||||
|
parser.error('Remote port missing.')
|
||||||
|
remote_host, remote_port = remote_host.split(':', 1)
|
||||||
|
try:
|
||||||
|
remote_port = int(remote_port)
|
||||||
|
except:
|
||||||
|
parser.error('Remote port must be a number.')
|
||||||
|
|
||||||
|
if not options.ssh_host:
|
||||||
|
parser.error('SSH host is required.')
|
||||||
|
if ':' in options.ssh_host:
|
||||||
|
options.ssh_host, options.ssh_port = options.ssh_host.split(':', 1)
|
||||||
|
try:
|
||||||
|
options.ssh_port = int(options.ssh_port)
|
||||||
|
except:
|
||||||
|
parser.error('SSH port must be a number.')
|
||||||
|
|
||||||
|
host_keys = load_host_keys()
|
||||||
|
if not host_keys.has_key(options.ssh_host):
|
||||||
|
print '*** Warning: no host key for %s' % options.ssh_host
|
||||||
|
expected_host_key_type = None
|
||||||
|
expected_host_key = None
|
||||||
|
else:
|
||||||
|
expected_host_key_type = host_keys[options.ssh_host].keys()[0]
|
||||||
|
expected_host_key = host_keys[options.ssh_host][expected_host_key_type]
|
||||||
|
|
||||||
|
key = None
|
||||||
|
password = None
|
||||||
|
if options.use_key:
|
||||||
|
try:
|
||||||
|
key = paramiko.RSAKey.from_private_key_file(options.keyfile)
|
||||||
|
except paramiko.PasswordRequiredException:
|
||||||
|
if not READPASS:
|
||||||
|
print '*** Password needed for keyfile (use -P): %s' % options.keyfile
|
||||||
|
sys.exit(1)
|
||||||
|
key_password = raw_input()
|
||||||
|
try:
|
||||||
|
key = paramiko.RSAKey.from_private_key_file(options.keyfile, key_password)
|
||||||
|
except:
|
||||||
|
print '*** Unable to read keyfile: %s' % options.keyfile
|
||||||
|
sys.exit(1)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
if key is None:
|
||||||
|
# try reading a password then
|
||||||
|
if not READPASS:
|
||||||
|
print '*** Either a valid private key or password is required (use -K or -P).'
|
||||||
|
sys.exit(1)
|
||||||
|
password = raw_input()
|
||||||
|
|
||||||
|
verbose('Connecting to ssh host %s:%d ...' % (options.ssh_host, options.ssh_port))
|
||||||
|
|
||||||
|
transport = paramiko.Transport((options.ssh_host, options.ssh_port))
|
||||||
|
transport.connect(hostkeytype=expected_host_key_type,
|
||||||
|
hostkey=expected_host_key,
|
||||||
|
username=options.user,
|
||||||
|
password=password,
|
||||||
|
pkey=key)
|
||||||
|
|
||||||
|
verbose('Now forwarding port %d to %s:%d ...' % (options.port, remote_host, remote_port))
|
||||||
|
|
||||||
|
try:
|
||||||
|
forward_tunnel(options.port, remote_host, remote_port, transport)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
print 'Port forwarding stopped.'
|
||||||
|
sys.exit(0)
|
|
@ -48,7 +48,7 @@ released under the GNU Lesser General Public License (LGPL).
|
||||||
|
|
||||||
Website: U{http://www.lag.net/~robey/paramiko/}
|
Website: U{http://www.lag.net/~robey/paramiko/}
|
||||||
|
|
||||||
@version: 0.9 (fearow)
|
@version: 0.9 (gyarados)
|
||||||
@author: Robey Pointer
|
@author: Robey Pointer
|
||||||
@contact: robey@lag.net
|
@contact: robey@lag.net
|
||||||
@license: GNU Lesser General Public License (LGPL)
|
@license: GNU Lesser General Public License (LGPL)
|
||||||
|
@ -61,9 +61,8 @@ if sys.version_info < (2, 2):
|
||||||
|
|
||||||
|
|
||||||
__author__ = "Robey Pointer <robey@lag.net>"
|
__author__ = "Robey Pointer <robey@lag.net>"
|
||||||
__date__ = "23 Apr 2004"
|
__date__ = "31 May 2004"
|
||||||
__version__ = "0.9-fearow"
|
__version__ = "0.9-gyarados"
|
||||||
#__credits__ = "Huzzah!"
|
|
||||||
__license__ = "GNU Lesser General Public License (LGPL)"
|
__license__ = "GNU Lesser General Public License (LGPL)"
|
||||||
|
|
||||||
|
|
||||||
|
|
4
setup.py
4
setup.py
|
@ -13,13 +13,13 @@ Required packages:
|
||||||
'''
|
'''
|
||||||
|
|
||||||
setup(name = "paramiko",
|
setup(name = "paramiko",
|
||||||
version = "0.9-fearow",
|
version = "0.9-gyarados",
|
||||||
description = "SSH2 protocol library",
|
description = "SSH2 protocol library",
|
||||||
author = "Robey Pointer",
|
author = "Robey Pointer",
|
||||||
author_email = "robey@lag.net",
|
author_email = "robey@lag.net",
|
||||||
url = "http://www.lag.net/~robey/paramiko/",
|
url = "http://www.lag.net/~robey/paramiko/",
|
||||||
packages = [ 'paramiko' ],
|
packages = [ 'paramiko' ],
|
||||||
download_url = 'http://www.lag.net/~robey/paramiko/paramiko-0.9-fearow.zip',
|
download_url = 'http://www.lag.net/~robey/paramiko/paramiko-0.9-gyarados.zip',
|
||||||
license = 'LGPL',
|
license = 'LGPL',
|
||||||
platforms = 'Posix; MacOS X; Windows',
|
platforms = 'Posix; MacOS X; Windows',
|
||||||
classifiers = [ 'Development Status :: 3 - Alpha',
|
classifiers = [ 'Development Status :: 3 - Alpha',
|
||||||
|
|
Loading…
Reference in New Issue