mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 10:26:02 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			352 lines
		
	
	
	
		
			9.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			352 lines
		
	
	
	
		
			9.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
| #! /usr/bin/env python
 | |
| 
 | |
| # A simple gopher client.
 | |
| #
 | |
| # Usage: gopher [ [selector] host [port] ]
 | |
| 
 | |
| import sys
 | |
| import os
 | |
| import socket
 | |
| 
 | |
| # Default selector, host and port
 | |
| DEF_SELECTOR = ''
 | |
| DEF_HOST     = 'gopher.micro.umn.edu'
 | |
| DEF_PORT     = 70
 | |
| 
 | |
| # Recognized file types
 | |
| T_TEXTFILE  = '0'
 | |
| T_MENU      = '1'
 | |
| T_CSO       = '2'
 | |
| T_ERROR     = '3'
 | |
| T_BINHEX    = '4'
 | |
| T_DOS       = '5'
 | |
| T_UUENCODE  = '6'
 | |
| T_SEARCH    = '7'
 | |
| T_TELNET    = '8'
 | |
| T_BINARY    = '9'
 | |
| T_REDUNDANT = '+'
 | |
| T_SOUND     = 's'
 | |
| 
 | |
| # Dictionary mapping types to strings
 | |
| typename = {'0': '<TEXT>', '1': '<DIR>', '2': '<CSO>', '3': '<ERROR>', \
 | |
|         '4': '<BINHEX>', '5': '<DOS>', '6': '<UUENCODE>', '7': '<SEARCH>', \
 | |
|         '8': '<TELNET>', '9': '<BINARY>', '+': '<REDUNDANT>', 's': '<SOUND>'}
 | |
| 
 | |
| # Oft-used characters and strings
 | |
| CRLF = '\r\n'
 | |
| TAB = '\t'
 | |
| 
 | |
| # Open a TCP connection to a given host and port
 | |
| def open_socket(host, port):
 | |
|     if not port:
 | |
|         port = DEF_PORT
 | |
|     elif type(port) == type(''):
 | |
|         port = int(port)
 | |
|     s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
 | |
|     s.connect((host, port))
 | |
|     return s
 | |
| 
 | |
| # Send a selector to a given host and port, return a file with the reply
 | |
| def send_request(selector, host, port):
 | |
|     s = open_socket(host, port)
 | |
|     s.send(selector + CRLF)
 | |
|     s.shutdown(1)
 | |
|     return s.makefile('r')
 | |
| 
 | |
| # Get a menu in the form of a list of entries
 | |
| def get_menu(selector, host, port):
 | |
|     f = send_request(selector, host, port)
 | |
|     list = []
 | |
|     while 1:
 | |
|         line = f.readline()
 | |
|         if not line:
 | |
|             print('(Unexpected EOF from server)')
 | |
|             break
 | |
|         if line[-2:] == CRLF:
 | |
|             line = line[:-2]
 | |
|         elif line[-1:] in CRLF:
 | |
|             line = line[:-1]
 | |
|         if line == '.':
 | |
|             break
 | |
|         if not line:
 | |
|             print('(Empty line from server)')
 | |
|             continue
 | |
|         typechar = line[0]
 | |
|         parts = line[1:].split(TAB)
 | |
|         if len(parts) < 4:
 | |
|             print('(Bad line from server: %r)' % (line,))
 | |
|             continue
 | |
|         if len(parts) > 4:
 | |
|             print('(Extra info from server: %r)' % (parts[4:],))
 | |
|         parts.insert(0, typechar)
 | |
|         list.append(parts)
 | |
|     f.close()
 | |
|     return list
 | |
| 
 | |
| # Get a text file as a list of lines, with trailing CRLF stripped
 | |
| def get_textfile(selector, host, port):
 | |
|     list = []
 | |
|     get_alt_textfile(selector, host, port, list.append)
 | |
|     return list
 | |
| 
 | |
| # Get a text file and pass each line to a function, with trailing CRLF stripped
 | |
| def get_alt_textfile(selector, host, port, func):
 | |
|     f = send_request(selector, host, port)
 | |
|     while 1:
 | |
|         line = f.readline()
 | |
|         if not line:
 | |
|             print('(Unexpected EOF from server)')
 | |
|             break
 | |
|         if line[-2:] == CRLF:
 | |
|             line = line[:-2]
 | |
|         elif line[-1:] in CRLF:
 | |
|             line = line[:-1]
 | |
|         if line == '.':
 | |
|             break
 | |
|         if line[:2] == '..':
 | |
|             line = line[1:]
 | |
|         func(line)
 | |
|     f.close()
 | |
| 
 | |
| # Get a binary file as one solid data block
 | |
| def get_binary(selector, host, port):
 | |
|     f = send_request(selector, host, port)
 | |
|     data = f.read()
 | |
|     f.close()
 | |
|     return data
 | |
| 
 | |
| # Get a binary file and pass each block to a function
 | |
| def get_alt_binary(selector, host, port, func, blocksize):
 | |
|     f = send_request(selector, host, port)
 | |
|     while 1:
 | |
|         data = f.read(blocksize)
 | |
|         if not data:
 | |
|             break
 | |
|         func(data)
 | |
| 
 | |
| # A *very* simple interactive browser
 | |
| 
 | |
| # Browser main command, has default arguments
 | |
| def browser(*args):
 | |
|     selector = DEF_SELECTOR
 | |
|     host = DEF_HOST
 | |
|     port = DEF_PORT
 | |
|     n = len(args)
 | |
|     if n > 0 and args[0]:
 | |
|         selector = args[0]
 | |
|     if n > 1 and args[1]:
 | |
|         host = args[1]
 | |
|     if n > 2 and args[2]:
 | |
|         port = args[2]
 | |
|     if n > 3:
 | |
|         raise RuntimeError('too many args')
 | |
|     try:
 | |
|         browse_menu(selector, host, port)
 | |
|     except socket.error as msg:
 | |
|         print('Socket error:', msg)
 | |
|         sys.exit(1)
 | |
|     except KeyboardInterrupt:
 | |
|         print('\n[Goodbye]')
 | |
| 
 | |
| # Browse a menu
 | |
| def browse_menu(selector, host, port):
 | |
|     list = get_menu(selector, host, port)
 | |
|     while 1:
 | |
|         print('----- MENU -----')
 | |
|         print('Selector:', repr(selector))
 | |
|         print('Host:', host, ' Port:', port)
 | |
|         print()
 | |
|         for i in range(len(list)):
 | |
|             item = list[i]
 | |
|             typechar, description = item[0], item[1]
 | |
|             print(repr(i+1).rjust(3) + ':', description, end=' ')
 | |
|             if typechar in typename:
 | |
|                 print(typename[typechar])
 | |
|             else:
 | |
|                 print('<TYPE=' + repr(typechar) + '>')
 | |
|         print()
 | |
|         while 1:
 | |
|             try:
 | |
|                 str = input('Choice [CR == up a level]: ')
 | |
|             except EOFError:
 | |
|                 print()
 | |
|                 return
 | |
|             if not str:
 | |
|                 return
 | |
|             try:
 | |
|                 choice = int(str)
 | |
|             except ValueError:
 | |
|                 print('Choice must be a number; try again:')
 | |
|                 continue
 | |
|             if not 0 < choice <= len(list):
 | |
|                 print('Choice out of range; try again:')
 | |
|                 continue
 | |
|             break
 | |
|         item = list[choice-1]
 | |
|         typechar = item[0]
 | |
|         [i_selector, i_host, i_port] = item[2:5]
 | |
|         if typechar in typebrowser:
 | |
|             browserfunc = typebrowser[typechar]
 | |
|             try:
 | |
|                 browserfunc(i_selector, i_host, i_port)
 | |
|             except (IOError, socket.error):
 | |
|                 t, v, tb = sys.exc_info()
 | |
|                 print('***', t, ':', v)
 | |
|         else:
 | |
|             print('Unsupported object type')
 | |
| 
 | |
| # Browse a text file
 | |
| def browse_textfile(selector, host, port):
 | |
|     x = None
 | |
|     try:
 | |
|         p = os.popen('${PAGER-more}', 'w')
 | |
|         x = SaveLines(p)
 | |
|         get_alt_textfile(selector, host, port, x.writeln)
 | |
|     except IOError as msg:
 | |
|         print('IOError:', msg)
 | |
|     if x:
 | |
|         x.close()
 | |
|     f = open_savefile()
 | |
|     if not f:
 | |
|         return
 | |
|     x = SaveLines(f)
 | |
|     try:
 | |
|         get_alt_textfile(selector, host, port, x.writeln)
 | |
|         print('Done.')
 | |
|     except IOError as msg:
 | |
|         print('IOError:', msg)
 | |
|     x.close()
 | |
| 
 | |
| def raw_input(prompt):
 | |
|     sys.stdout.write(prompt)
 | |
|     sys.stdout.flush()
 | |
|     return sys.stdin.readline()
 | |
| 
 | |
| # Browse a search index
 | |
| def browse_search(selector, host, port):
 | |
|     while 1:
 | |
|         print('----- SEARCH -----')
 | |
|         print('Selector:', repr(selector))
 | |
|         print('Host:', host, ' Port:', port)
 | |
|         print()
 | |
|         try:
 | |
|             query = input('Query [CR == up a level]: ')
 | |
|         except EOFError:
 | |
|             print()
 | |
|             break
 | |
|         query = query.strip()
 | |
|         if not query:
 | |
|             break
 | |
|         if '\t' in query:
 | |
|             print('Sorry, queries cannot contain tabs')
 | |
|             continue
 | |
|         browse_menu(selector + TAB + query, host, port)
 | |
| 
 | |
| # "Browse" telnet-based information, i.e. open a telnet session
 | |
| def browse_telnet(selector, host, port):
 | |
|     if selector:
 | |
|         print('Log in as', repr(selector))
 | |
|     if type(port) != type(''):
 | |
|         port = repr(port)
 | |
|     sts = os.system('set -x; exec telnet ' + host + ' ' + port)
 | |
|     if sts:
 | |
|         print('Exit status:', sts)
 | |
| 
 | |
| # "Browse" a binary file, i.e. save it to a file
 | |
| def browse_binary(selector, host, port):
 | |
|     f = open_savefile()
 | |
|     if not f:
 | |
|         return
 | |
|     x = SaveWithProgress(f)
 | |
|     get_alt_binary(selector, host, port, x.write, 8*1024)
 | |
|     x.close()
 | |
| 
 | |
| # "Browse" a sound file, i.e. play it or save it
 | |
| def browse_sound(selector, host, port):
 | |
|     browse_binary(selector, host, port)
 | |
| 
 | |
| # Dictionary mapping types to browser functions
 | |
| typebrowser = {'0': browse_textfile, '1': browse_menu, \
 | |
|         '4': browse_binary, '5': browse_binary, '6': browse_textfile, \
 | |
|         '7': browse_search, \
 | |
|         '8': browse_telnet, '9': browse_binary, 's': browse_sound}
 | |
| 
 | |
| # Class used to save lines, appending a newline to each line
 | |
| class SaveLines:
 | |
|     def __init__(self, f):
 | |
|         self.f = f
 | |
|     def writeln(self, line):
 | |
|         self.f.write(line + '\n')
 | |
|     def close(self):
 | |
|         sts = self.f.close()
 | |
|         if sts:
 | |
|             print('Exit status:', sts)
 | |
| 
 | |
| # Class used to save data while showing progress
 | |
| class SaveWithProgress:
 | |
|     def __init__(self, f):
 | |
|         self.f = f
 | |
|     def write(self, data):
 | |
|         sys.stdout.write('#')
 | |
|         sys.stdout.flush()
 | |
|         self.f.write(data)
 | |
|     def close(self):
 | |
|         print()
 | |
|         sts = self.f.close()
 | |
|         if sts:
 | |
|             print('Exit status:', sts)
 | |
| 
 | |
| # Ask for and open a save file, or return None if not to save
 | |
| def open_savefile():
 | |
|     try:
 | |
|         savefile = input( \
 | |
|     'Save as file [CR == don\'t save; |pipeline or ~user/... OK]: ')
 | |
|     except EOFError:
 | |
|         print()
 | |
|         return None
 | |
|     savefile = savefile.strip()
 | |
|     if not savefile:
 | |
|         return None
 | |
|     if savefile[0] == '|':
 | |
|         cmd = savefile[1:].strip()
 | |
|         try:
 | |
|             p = os.popen(cmd, 'w')
 | |
|         except IOError as msg:
 | |
|             print(repr(cmd), ':', msg)
 | |
|             return None
 | |
|         print('Piping through', repr(cmd), '...')
 | |
|         return p
 | |
|     if savefile[0] == '~':
 | |
|         savefile = os.path.expanduser(savefile)
 | |
|     try:
 | |
|         f = open(savefile, 'w')
 | |
|     except IOError as msg:
 | |
|         print(repr(savefile), ':', msg)
 | |
|         return None
 | |
|     print('Saving to', repr(savefile), '...')
 | |
|     return f
 | |
| 
 | |
| # Test program
 | |
| def test():
 | |
|     if sys.argv[4:]:
 | |
|         print('usage: gopher [ [selector] host [port] ]')
 | |
|         sys.exit(2)
 | |
|     elif sys.argv[3:]:
 | |
|         browser(sys.argv[1], sys.argv[2], sys.argv[3])
 | |
|     elif sys.argv[2:]:
 | |
|         try:
 | |
|             port = int(sys.argv[2])
 | |
|             selector = ''
 | |
|             host = sys.argv[1]
 | |
|         except ValueError:
 | |
|             selector = sys.argv[1]
 | |
|             host = sys.argv[2]
 | |
|             port = ''
 | |
|         browser(selector, host, port)
 | |
|     elif sys.argv[1:]:
 | |
|         browser('', sys.argv[1])
 | |
|     else:
 | |
|         browser()
 | |
| 
 | |
| # Call the test program as a main program
 | |
| test()
 | 
