mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 03:04:41 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1115 lines
		
	
	
	
		
			41 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			1115 lines
		
	
	
	
		
			41 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """An NNTP client class based on:
 | |
| - RFC 977: Network News Transfer Protocol
 | |
| - RFC 2980: Common NNTP Extensions
 | |
| - RFC 3977: Network News Transfer Protocol (version 2)
 | |
| 
 | |
| Example:
 | |
| 
 | |
| >>> from nntplib import NNTP
 | |
| >>> s = NNTP('news')
 | |
| >>> resp, count, first, last, name = s.group('comp.lang.python')
 | |
| >>> print('Group', name, 'has', count, 'articles, range', first, 'to', last)
 | |
| Group comp.lang.python has 51 articles, range 5770 to 5821
 | |
| >>> resp, subs = s.xhdr('subject', '{0}-{1}'.format(first, last))
 | |
| >>> resp = s.quit()
 | |
| >>>
 | |
| 
 | |
| Here 'resp' is the server response line.
 | |
| Error responses are turned into exceptions.
 | |
| 
 | |
| To post an article from a file:
 | |
| >>> f = open(filename, 'rb') # file containing article, including header
 | |
| >>> resp = s.post(f)
 | |
| >>>
 | |
| 
 | |
| For descriptions of all methods, read the comments in the code below.
 | |
| Note that all arguments and return values representing article numbers
 | |
| are strings, not numbers, since they are rarely used for calculations.
 | |
| """
 | |
| 
 | |
| # RFC 977 by Brian Kantor and Phil Lapsley.
 | |
| # xover, xgtitle, xpath, date methods by Kevan Heydon
 | |
| 
 | |
| # Incompatible changes from the 2.x nntplib:
 | |
| # - all commands are encoded as UTF-8 data (using the "surrogateescape"
 | |
| #   error handler), except for raw message data (POST, IHAVE)
 | |
| # - all responses are decoded as UTF-8 data (using the "surrogateescape"
 | |
| #   error handler), except for raw message data (ARTICLE, HEAD, BODY)
 | |
| # - the `file` argument to various methods is keyword-only
 | |
| #
 | |
| # - NNTP.date() returns a datetime object
 | |
| # - NNTP.newgroups() and NNTP.newnews() take a datetime (or date) object,
 | |
| #   rather than a pair of (date, time) strings.
 | |
| # - NNTP.newgroups() and NNTP.list() return a list of GroupInfo named tuples
 | |
| # - NNTP.descriptions() returns a dict mapping group names to descriptions
 | |
| # - NNTP.xover() returns a list of dicts mapping field names (header or metadata)
 | |
| #   to field values; each dict representing a message overview.
 | |
| # - NNTP.article(), NNTP.head() and NNTP.body() return a (response, ArticleInfo)
 | |
| #   tuple.
 | |
| # - the "internal" methods have been marked private (they now start with
 | |
| #   an underscore)
 | |
| 
 | |
| # Other changes from the 2.x/3.1 nntplib:
 | |
| # - automatic querying of capabilities at connect
 | |
| # - New method NNTP.getcapabilities()
 | |
| # - New method NNTP.over()
 | |
| # - New helper function decode_header()
 | |
| # - NNTP.post() and NNTP.ihave() accept file objects, bytes-like objects and
 | |
| #   arbitrary iterables yielding lines.
 | |
| # - An extensive test suite :-)
 | |
| 
 | |
| # TODO:
 | |
| # - return structured data (GroupInfo etc.) everywhere
 | |
| # - support HDR
 | |
| 
 | |
| # Imports
 | |
| import re
 | |
| import socket
 | |
| import collections
 | |
| import datetime
 | |
| import warnings
 | |
| 
 | |
| try:
 | |
|     import ssl
 | |
| except ImportError:
 | |
|     _have_ssl = False
 | |
| else:
 | |
|     _have_ssl = True
 | |
| 
 | |
| from email.header import decode_header as _email_decode_header
 | |
| from socket import _GLOBAL_DEFAULT_TIMEOUT
 | |
| 
 | |
| __all__ = ["NNTP",
 | |
|            "NNTPReplyError", "NNTPTemporaryError", "NNTPPermanentError",
 | |
|            "NNTPProtocolError", "NNTPDataError",
 | |
|            "decode_header",
 | |
|            ]
 | |
| 
 | |
| # Exceptions raised when an error or invalid response is received
 | |
| class NNTPError(Exception):
 | |
|     """Base class for all nntplib exceptions"""
 | |
|     def __init__(self, *args):
 | |
|         Exception.__init__(self, *args)
 | |
|         try:
 | |
|             self.response = args[0]
 | |
|         except IndexError:
 | |
|             self.response = 'No response given'
 | |
| 
 | |
| class NNTPReplyError(NNTPError):
 | |
|     """Unexpected [123]xx reply"""
 | |
|     pass
 | |
| 
 | |
| class NNTPTemporaryError(NNTPError):
 | |
|     """4xx errors"""
 | |
|     pass
 | |
| 
 | |
| class NNTPPermanentError(NNTPError):
 | |
|     """5xx errors"""
 | |
|     pass
 | |
| 
 | |
| class NNTPProtocolError(NNTPError):
 | |
|     """Response does not begin with [1-5]"""
 | |
|     pass
 | |
| 
 | |
| class NNTPDataError(NNTPError):
 | |
|     """Error in response data"""
 | |
|     pass
 | |
| 
 | |
| 
 | |
| # Standard port used by NNTP servers
 | |
| NNTP_PORT = 119
 | |
| NNTP_SSL_PORT = 563
 | |
| 
 | |
| # Response numbers that are followed by additional text (e.g. article)
 | |
| _LONGRESP = {
 | |
|     '100',   # HELP
 | |
|     '101',   # CAPABILITIES
 | |
|     '211',   # LISTGROUP   (also not multi-line with GROUP)
 | |
|     '215',   # LIST
 | |
|     '220',   # ARTICLE
 | |
|     '221',   # HEAD, XHDR
 | |
|     '222',   # BODY
 | |
|     '224',   # OVER, XOVER
 | |
|     '225',   # HDR
 | |
|     '230',   # NEWNEWS
 | |
|     '231',   # NEWGROUPS
 | |
|     '282',   # XGTITLE
 | |
| }
 | |
| 
 | |
| # Default decoded value for LIST OVERVIEW.FMT if not supported
 | |
| _DEFAULT_OVERVIEW_FMT = [
 | |
|     "subject", "from", "date", "message-id", "references", ":bytes", ":lines"]
 | |
| 
 | |
| # Alternative names allowed in LIST OVERVIEW.FMT response
 | |
| _OVERVIEW_FMT_ALTERNATIVES = {
 | |
|     'bytes': ':bytes',
 | |
|     'lines': ':lines',
 | |
| }
 | |
| 
 | |
| # Line terminators (we always output CRLF, but accept any of CRLF, CR, LF)
 | |
| _CRLF = b'\r\n'
 | |
| 
 | |
| GroupInfo = collections.namedtuple('GroupInfo',
 | |
|                                    ['group', 'last', 'first', 'flag'])
 | |
| 
 | |
| ArticleInfo = collections.namedtuple('ArticleInfo',
 | |
|                                      ['number', 'message_id', 'lines'])
 | |
| 
 | |
| 
 | |
| # Helper function(s)
 | |
| def decode_header(header_str):
 | |
|     """Takes an unicode string representing a munged header value
 | |
|     and decodes it as a (possibly non-ASCII) readable value."""
 | |
|     parts = []
 | |
|     for v, enc in _email_decode_header(header_str):
 | |
|         if isinstance(v, bytes):
 | |
|             parts.append(v.decode(enc or 'ascii'))
 | |
|         else:
 | |
|             parts.append(v)
 | |
|     return ' '.join(parts)
 | |
| 
 | |
| def _parse_overview_fmt(lines):
 | |
|     """Parse a list of string representing the response to LIST OVERVIEW.FMT
 | |
|     and return a list of header/metadata names.
 | |
|     Raises NNTPDataError if the response is not compliant
 | |
|     (cf. RFC 3977, section 8.4)."""
 | |
|     fmt = []
 | |
|     for line in lines:
 | |
|         if line[0] == ':':
 | |
|             # Metadata name (e.g. ":bytes")
 | |
|             name, _, suffix = line[1:].partition(':')
 | |
|             name = ':' + name
 | |
|         else:
 | |
|             # Header name (e.g. "Subject:" or "Xref:full")
 | |
|             name, _, suffix = line.partition(':')
 | |
|         name = name.lower()
 | |
|         name = _OVERVIEW_FMT_ALTERNATIVES.get(name, name)
 | |
|         # Should we do something with the suffix?
 | |
|         fmt.append(name)
 | |
|     defaults = _DEFAULT_OVERVIEW_FMT
 | |
|     if len(fmt) < len(defaults):
 | |
|         raise NNTPDataError("LIST OVERVIEW.FMT response too short")
 | |
|     if fmt[:len(defaults)] != defaults:
 | |
|         raise NNTPDataError("LIST OVERVIEW.FMT redefines default fields")
 | |
|     return fmt
 | |
| 
 | |
| def _parse_overview(lines, fmt, data_process_func=None):
 | |
|     """Parse the response to a OVER or XOVER command according to the
 | |
|     overview format `fmt`."""
 | |
|     n_defaults = len(_DEFAULT_OVERVIEW_FMT)
 | |
|     overview = []
 | |
|     for line in lines:
 | |
|         fields = {}
 | |
|         article_number, *tokens = line.split('\t')
 | |
|         article_number = int(article_number)
 | |
|         for i, token in enumerate(tokens):
 | |
|             if i >= len(fmt):
 | |
|                 # XXX should we raise an error? Some servers might not
 | |
|                 # support LIST OVERVIEW.FMT and still return additional
 | |
|                 # headers.
 | |
|                 continue
 | |
|             field_name = fmt[i]
 | |
|             is_metadata = field_name.startswith(':')
 | |
|             if i >= n_defaults and not is_metadata:
 | |
|                 # Non-default header names are included in full in the response
 | |
|                 # (unless the field is totally empty)
 | |
|                 h = field_name + ": "
 | |
|                 if token and token[:len(h)].lower() != h:
 | |
|                     raise NNTPDataError("OVER/XOVER response doesn't include "
 | |
|                                         "names of additional headers")
 | |
|                 token = token[len(h):] if token else None
 | |
|             fields[fmt[i]] = token
 | |
|         overview.append((article_number, fields))
 | |
|     return overview
 | |
| 
 | |
| def _parse_datetime(date_str, time_str=None):
 | |
|     """Parse a pair of (date, time) strings, and return a datetime object.
 | |
|     If only the date is given, it is assumed to be date and time
 | |
|     concatenated together (e.g. response to the DATE command).
 | |
|     """
 | |
|     if time_str is None:
 | |
|         time_str = date_str[-6:]
 | |
|         date_str = date_str[:-6]
 | |
|     hours = int(time_str[:2])
 | |
|     minutes = int(time_str[2:4])
 | |
|     seconds = int(time_str[4:])
 | |
|     year = int(date_str[:-4])
 | |
|     month = int(date_str[-4:-2])
 | |
|     day = int(date_str[-2:])
 | |
|     # RFC 3977 doesn't say how to interpret 2-char years.  Assume that
 | |
|     # there are no dates before 1970 on Usenet.
 | |
|     if year < 70:
 | |
|         year += 2000
 | |
|     elif year < 100:
 | |
|         year += 1900
 | |
|     return datetime.datetime(year, month, day, hours, minutes, seconds)
 | |
| 
 | |
| def _unparse_datetime(dt, legacy=False):
 | |
|     """Format a date or datetime object as a pair of (date, time) strings
 | |
|     in the format required by the NEWNEWS and NEWGROUPS commands.  If a
 | |
|     date object is passed, the time is assumed to be midnight (00h00).
 | |
| 
 | |
|     The returned representation depends on the legacy flag:
 | |
|     * if legacy is False (the default):
 | |
|       date has the YYYYMMDD format and time the HHMMSS format
 | |
|     * if legacy is True:
 | |
|       date has the YYMMDD format and time the HHMMSS format.
 | |
|     RFC 3977 compliant servers should understand both formats; therefore,
 | |
|     legacy is only needed when talking to old servers.
 | |
|     """
 | |
|     if not isinstance(dt, datetime.datetime):
 | |
|         time_str = "000000"
 | |
|     else:
 | |
|         time_str = "{0.hour:02d}{0.minute:02d}{0.second:02d}".format(dt)
 | |
|     y = dt.year
 | |
|     if legacy:
 | |
|         y = y % 100
 | |
|         date_str = "{0:02d}{1.month:02d}{1.day:02d}".format(y, dt)
 | |
|     else:
 | |
|         date_str = "{0:04d}{1.month:02d}{1.day:02d}".format(y, dt)
 | |
|     return date_str, time_str
 | |
| 
 | |
| 
 | |
| if _have_ssl:
 | |
| 
 | |
|     def _encrypt_on(sock, context):
 | |
|         """Wrap a socket in SSL/TLS. Arguments:
 | |
|         - sock: Socket to wrap
 | |
|         - context: SSL context to use for the encrypted connection
 | |
|         Returns:
 | |
|         - sock: New, encrypted socket.
 | |
|         """
 | |
|         # Generate a default SSL context if none was passed.
 | |
|         if context is None:
 | |
|             context = ssl.SSLContext(ssl.PROTOCOL_SSLv23)
 | |
|             # SSLv2 considered harmful.
 | |
|             context.options |= ssl.OP_NO_SSLv2
 | |
|         return context.wrap_socket(sock)
 | |
| 
 | |
| 
 | |
| # The classes themselves
 | |
| class _NNTPBase:
 | |
|     # UTF-8 is the character set for all NNTP commands and responses: they
 | |
|     # are automatically encoded (when sending) and decoded (and receiving)
 | |
|     # by this class.
 | |
|     # However, some multi-line data blocks can contain arbitrary bytes (for
 | |
|     # example, latin-1 or utf-16 data in the body of a message). Commands
 | |
|     # taking (POST, IHAVE) or returning (HEAD, BODY, ARTICLE) raw message
 | |
|     # data will therefore only accept and produce bytes objects.
 | |
|     # Furthermore, since there could be non-compliant servers out there,
 | |
|     # we use 'surrogateescape' as the error handler for fault tolerance
 | |
|     # and easy round-tripping. This could be useful for some applications
 | |
|     # (e.g. NNTP gateways).
 | |
| 
 | |
|     encoding = 'utf-8'
 | |
|     errors = 'surrogateescape'
 | |
| 
 | |
|     def __init__(self, file, host,
 | |
|                  readermode=None, timeout=_GLOBAL_DEFAULT_TIMEOUT):
 | |
|         """Initialize an instance.  Arguments:
 | |
|         - file: file-like object (open for read/write in binary mode)
 | |
|         - host: hostname of the server
 | |
|         - readermode: if true, send 'mode reader' command after
 | |
|                       connecting.
 | |
|         - timeout: timeout (in seconds) used for socket connections
 | |
| 
 | |
|         readermode is sometimes necessary if you are connecting to an
 | |
|         NNTP server on the local machine and intend to call
 | |
|         reader-specific commands, such as `group'.  If you get
 | |
|         unexpected NNTPPermanentErrors, you might need to set
 | |
|         readermode.
 | |
|         """
 | |
|         self.host = host
 | |
|         self.file = file
 | |
|         self.debugging = 0
 | |
|         self.welcome = self._getresp()
 | |
| 
 | |
|         # 'MODE READER' is sometimes necessary to enable 'reader' mode.
 | |
|         # However, the order in which 'MODE READER' and 'AUTHINFO' need to
 | |
|         # arrive differs between some NNTP servers. If _setreadermode() fails
 | |
|         # with an authorization failed error, it will set this to True;
 | |
|         # the login() routine will interpret that as a request to try again
 | |
|         # after performing its normal function.
 | |
|         self.readermode_afterauth = False
 | |
|         if readermode:
 | |
|             self._setreadermode()
 | |
| 
 | |
|         # RFC 4642 2.2.2: Both the client and the server MUST know if there is
 | |
|         # a TLS session active.  A client MUST NOT attempt to start a TLS
 | |
|         # session if a TLS session is already active.
 | |
|         self.tls_on = False
 | |
| 
 | |
|         # Inquire about capabilities (RFC 3977).
 | |
|         self._caps = None
 | |
|         self.getcapabilities()
 | |
| 
 | |
|         # Log in and encryption setup order is left to subclasses.
 | |
|         self.authenticated = False
 | |
| 
 | |
|     def __enter__(self):
 | |
|         return self
 | |
| 
 | |
|     def __exit__(self, *args):
 | |
|         is_connected = lambda: hasattr(self, "file")
 | |
|         if is_connected():
 | |
|             try:
 | |
|                 self.quit()
 | |
|             except (socket.error, EOFError):
 | |
|                 pass
 | |
|             finally:
 | |
|                 if is_connected():
 | |
|                     self._close()
 | |
| 
 | |
|     def getwelcome(self):
 | |
|         """Get the welcome message from the server
 | |
|         (this is read and squirreled away by __init__()).
 | |
|         If the response code is 200, posting is allowed;
 | |
|         if it 201, posting is not allowed."""
 | |
| 
 | |
|         if self.debugging: print('*welcome*', repr(self.welcome))
 | |
|         return self.welcome
 | |
| 
 | |
|     def getcapabilities(self):
 | |
|         """Get the server capabilities, as read by __init__().
 | |
|         If the CAPABILITIES command is not supported, an empty dict is
 | |
|         returned."""
 | |
|         if self._caps is None:
 | |
|             self.nntp_version = 1
 | |
|             self.nntp_implementation = None
 | |
|             try:
 | |
|                 resp, caps = self.capabilities()
 | |
|             except NNTPPermanentError:
 | |
|                 # Server doesn't support capabilities
 | |
|                 self._caps = {}
 | |
|             else:
 | |
|                 self._caps = caps
 | |
|                 if 'VERSION' in caps:
 | |
|                     # The server can advertise several supported versions,
 | |
|                     # choose the highest.
 | |
|                     self.nntp_version = max(map(int, caps['VERSION']))
 | |
|                 if 'IMPLEMENTATION' in caps:
 | |
|                     self.nntp_implementation = ' '.join(caps['IMPLEMENTATION'])
 | |
|         return self._caps
 | |
| 
 | |
|     def set_debuglevel(self, level):
 | |
|         """Set the debugging level.  Argument 'level' means:
 | |
|         0: no debugging output (default)
 | |
|         1: print commands and responses but not body text etc.
 | |
|         2: also print raw lines read and sent before stripping CR/LF"""
 | |
| 
 | |
|         self.debugging = level
 | |
|     debug = set_debuglevel
 | |
| 
 | |
|     def _putline(self, line):
 | |
|         """Internal: send one line to the server, appending CRLF.
 | |
|         The `line` must be a bytes-like object."""
 | |
|         line = line + _CRLF
 | |
|         if self.debugging > 1: print('*put*', repr(line))
 | |
|         self.file.write(line)
 | |
|         self.file.flush()
 | |
| 
 | |
|     def _putcmd(self, line):
 | |
|         """Internal: send one command to the server (through _putline()).
 | |
|         The `line` must be an unicode string."""
 | |
|         if self.debugging: print('*cmd*', repr(line))
 | |
|         line = line.encode(self.encoding, self.errors)
 | |
|         self._putline(line)
 | |
| 
 | |
|     def _getline(self, strip_crlf=True):
 | |
|         """Internal: return one line from the server, stripping _CRLF.
 | |
|         Raise EOFError if the connection is closed.
 | |
|         Returns a bytes object."""
 | |
|         line = self.file.readline()
 | |
|         if self.debugging > 1:
 | |
|             print('*get*', repr(line))
 | |
|         if not line: raise EOFError
 | |
|         if strip_crlf:
 | |
|             if line[-2:] == _CRLF:
 | |
|                 line = line[:-2]
 | |
|             elif line[-1:] in _CRLF:
 | |
|                 line = line[:-1]
 | |
|         return line
 | |
| 
 | |
|     def _getresp(self):
 | |
|         """Internal: get a response from the server.
 | |
|         Raise various errors if the response indicates an error.
 | |
|         Returns an unicode string."""
 | |
|         resp = self._getline()
 | |
|         if self.debugging: print('*resp*', repr(resp))
 | |
|         resp = resp.decode(self.encoding, self.errors)
 | |
|         c = resp[:1]
 | |
|         if c == '4':
 | |
|             raise NNTPTemporaryError(resp)
 | |
|         if c == '5':
 | |
|             raise NNTPPermanentError(resp)
 | |
|         if c not in '123':
 | |
|             raise NNTPProtocolError(resp)
 | |
|         return resp
 | |
| 
 | |
|     def _getlongresp(self, file=None):
 | |
|         """Internal: get a response plus following text from the server.
 | |
|         Raise various errors if the response indicates an error.
 | |
| 
 | |
|         Returns a (response, lines) tuple where `response` is an unicode
 | |
|         string and `lines` is a list of bytes objects.
 | |
|         If `file` is a file-like object, it must be open in binary mode.
 | |
|         """
 | |
| 
 | |
|         openedFile = None
 | |
|         try:
 | |
|             # If a string was passed then open a file with that name
 | |
|             if isinstance(file, (str, bytes)):
 | |
|                 openedFile = file = open(file, "wb")
 | |
| 
 | |
|             resp = self._getresp()
 | |
|             if resp[:3] not in _LONGRESP:
 | |
|                 raise NNTPReplyError(resp)
 | |
| 
 | |
|             lines = []
 | |
|             if file is not None:
 | |
|                 # XXX lines = None instead?
 | |
|                 terminators = (b'.' + _CRLF, b'.\n')
 | |
|                 while 1:
 | |
|                     line = self._getline(False)
 | |
|                     if line in terminators:
 | |
|                         break
 | |
|                     if line.startswith(b'..'):
 | |
|                         line = line[1:]
 | |
|                     file.write(line)
 | |
|             else:
 | |
|                 terminator = b'.'
 | |
|                 while 1:
 | |
|                     line = self._getline()
 | |
|                     if line == terminator:
 | |
|                         break
 | |
|                     if line.startswith(b'..'):
 | |
|                         line = line[1:]
 | |
|                     lines.append(line)
 | |
|         finally:
 | |
|             # If this method created the file, then it must close it
 | |
|             if openedFile:
 | |
|                 openedFile.close()
 | |
| 
 | |
|         return resp, lines
 | |
| 
 | |
|     def _shortcmd(self, line):
 | |
|         """Internal: send a command and get the response.
 | |
|         Same return value as _getresp()."""
 | |
|         self._putcmd(line)
 | |
|         return self._getresp()
 | |
| 
 | |
|     def _longcmd(self, line, file=None):
 | |
|         """Internal: send a command and get the response plus following text.
 | |
|         Same return value as _getlongresp()."""
 | |
|         self._putcmd(line)
 | |
|         return self._getlongresp(file)
 | |
| 
 | |
|     def _longcmdstring(self, line, file=None):
 | |
|         """Internal: send a command and get the response plus following text.
 | |
|         Same as _longcmd() and _getlongresp(), except that the returned `lines`
 | |
|         are unicode strings rather than bytes objects.
 | |
|         """
 | |
|         self._putcmd(line)
 | |
|         resp, list = self._getlongresp(file)
 | |
|         return resp, [line.decode(self.encoding, self.errors)
 | |
|                       for line in list]
 | |
| 
 | |
|     def _getoverviewfmt(self):
 | |
|         """Internal: get the overview format. Queries the server if not
 | |
|         already done, else returns the cached value."""
 | |
|         try:
 | |
|             return self._cachedoverviewfmt
 | |
|         except AttributeError:
 | |
|             pass
 | |
|         try:
 | |
|             resp, lines = self._longcmdstring("LIST OVERVIEW.FMT")
 | |
|         except NNTPPermanentError:
 | |
|             # Not supported by server?
 | |
|             fmt = _DEFAULT_OVERVIEW_FMT[:]
 | |
|         else:
 | |
|             fmt = _parse_overview_fmt(lines)
 | |
|         self._cachedoverviewfmt = fmt
 | |
|         return fmt
 | |
| 
 | |
|     def _grouplist(self, lines):
 | |
|         # Parse lines into "group last first flag"
 | |
|         return [GroupInfo(*line.split()) for line in lines]
 | |
| 
 | |
|     def capabilities(self):
 | |
|         """Process a CAPABILITIES command.  Not supported by all servers.
 | |
|         Return:
 | |
|         - resp: server response if successful
 | |
|         - caps: a dictionary mapping capability names to lists of tokens
 | |
|         (for example {'VERSION': ['2'], 'OVER': [], LIST: ['ACTIVE', 'HEADERS'] })
 | |
|         """
 | |
|         caps = {}
 | |
|         resp, lines = self._longcmdstring("CAPABILITIES")
 | |
|         for line in lines:
 | |
|             name, *tokens = line.split()
 | |
|             caps[name] = tokens
 | |
|         return resp, caps
 | |
| 
 | |
|     def newgroups(self, date, *, file=None):
 | |
|         """Process a NEWGROUPS command.  Arguments:
 | |
|         - date: a date or datetime object
 | |
|         Return:
 | |
|         - resp: server response if successful
 | |
|         - list: list of newsgroup names
 | |
|         """
 | |
|         if not isinstance(date, (datetime.date, datetime.date)):
 | |
|             raise TypeError(
 | |
|                 "the date parameter must be a date or datetime object, "
 | |
|                 "not '{:40}'".format(date.__class__.__name__))
 | |
|         date_str, time_str = _unparse_datetime(date, self.nntp_version < 2)
 | |
|         cmd = 'NEWGROUPS {0} {1}'.format(date_str, time_str)
 | |
|         resp, lines = self._longcmdstring(cmd, file)
 | |
|         return resp, self._grouplist(lines)
 | |
| 
 | |
|     def newnews(self, group, date, *, file=None):
 | |
|         """Process a NEWNEWS command.  Arguments:
 | |
|         - group: group name or '*'
 | |
|         - date: a date or datetime object
 | |
|         Return:
 | |
|         - resp: server response if successful
 | |
|         - list: list of message ids
 | |
|         """
 | |
|         if not isinstance(date, (datetime.date, datetime.date)):
 | |
|             raise TypeError(
 | |
|                 "the date parameter must be a date or datetime object, "
 | |
|                 "not '{:40}'".format(date.__class__.__name__))
 | |
|         date_str, time_str = _unparse_datetime(date, self.nntp_version < 2)
 | |
|         cmd = 'NEWNEWS {0} {1} {2}'.format(group, date_str, time_str)
 | |
|         return self._longcmdstring(cmd, file)
 | |
| 
 | |
|     def list(self, group_pattern=None, *, file=None):
 | |
|         """Process a LIST or LIST ACTIVE command. Arguments:
 | |
|         - group_pattern: a pattern indicating which groups to query
 | |
|         - file: Filename string or file object to store the result in
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - list: list of (group, last, first, flag) (strings)
 | |
|         """
 | |
|         if group_pattern is not None:
 | |
|             command = 'LIST ACTIVE ' + group_pattern
 | |
|         else:
 | |
|             command = 'LIST'
 | |
|         resp, lines = self._longcmdstring(command, file)
 | |
|         return resp, self._grouplist(lines)
 | |
| 
 | |
|     def _getdescriptions(self, group_pattern, return_all):
 | |
|         line_pat = re.compile('^(?P<group>[^ \t]+)[ \t]+(.*)$')
 | |
|         # Try the more std (acc. to RFC2980) LIST NEWSGROUPS first
 | |
|         resp, lines = self._longcmdstring('LIST NEWSGROUPS ' + group_pattern)
 | |
|         if not resp.startswith('215'):
 | |
|             # Now the deprecated XGTITLE.  This either raises an error
 | |
|             # or succeeds with the same output structure as LIST
 | |
|             # NEWSGROUPS.
 | |
|             resp, lines = self._longcmdstring('XGTITLE ' + group_pattern)
 | |
|         groups = {}
 | |
|         for raw_line in lines:
 | |
|             match = line_pat.search(raw_line.strip())
 | |
|             if match:
 | |
|                 name, desc = match.group(1, 2)
 | |
|                 if not return_all:
 | |
|                     return desc
 | |
|                 groups[name] = desc
 | |
|         if return_all:
 | |
|             return resp, groups
 | |
|         else:
 | |
|             # Nothing found
 | |
|             return ''
 | |
| 
 | |
|     def description(self, group):
 | |
|         """Get a description for a single group.  If more than one
 | |
|         group matches ('group' is a pattern), return the first.  If no
 | |
|         group matches, return an empty string.
 | |
| 
 | |
|         This elides the response code from the server, since it can
 | |
|         only be '215' or '285' (for xgtitle) anyway.  If the response
 | |
|         code is needed, use the 'descriptions' method.
 | |
| 
 | |
|         NOTE: This neither checks for a wildcard in 'group' nor does
 | |
|         it check whether the group actually exists."""
 | |
|         return self._getdescriptions(group, False)
 | |
| 
 | |
|     def descriptions(self, group_pattern):
 | |
|         """Get descriptions for a range of groups."""
 | |
|         return self._getdescriptions(group_pattern, True)
 | |
| 
 | |
|     def group(self, name):
 | |
|         """Process a GROUP command.  Argument:
 | |
|         - group: the group name
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - count: number of articles
 | |
|         - first: first article number
 | |
|         - last: last article number
 | |
|         - name: the group name
 | |
|         """
 | |
|         resp = self._shortcmd('GROUP ' + name)
 | |
|         if not resp.startswith('211'):
 | |
|             raise NNTPReplyError(resp)
 | |
|         words = resp.split()
 | |
|         count = first = last = 0
 | |
|         n = len(words)
 | |
|         if n > 1:
 | |
|             count = words[1]
 | |
|             if n > 2:
 | |
|                 first = words[2]
 | |
|                 if n > 3:
 | |
|                     last = words[3]
 | |
|                     if n > 4:
 | |
|                         name = words[4].lower()
 | |
|         return resp, int(count), int(first), int(last), name
 | |
| 
 | |
|     def help(self, *, file=None):
 | |
|         """Process a HELP command. Argument:
 | |
|         - file: Filename string or file object to store the result in
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - list: list of strings returned by the server in response to the
 | |
|                 HELP command
 | |
|         """
 | |
|         return self._longcmdstring('HELP', file)
 | |
| 
 | |
|     def _statparse(self, resp):
 | |
|         """Internal: parse the response line of a STAT, NEXT, LAST,
 | |
|         ARTICLE, HEAD or BODY command."""
 | |
|         if not resp.startswith('22'):
 | |
|             raise NNTPReplyError(resp)
 | |
|         words = resp.split()
 | |
|         art_num = int(words[1])
 | |
|         message_id = words[2]
 | |
|         return resp, art_num, message_id
 | |
| 
 | |
|     def _statcmd(self, line):
 | |
|         """Internal: process a STAT, NEXT or LAST command."""
 | |
|         resp = self._shortcmd(line)
 | |
|         return self._statparse(resp)
 | |
| 
 | |
|     def stat(self, message_spec=None):
 | |
|         """Process a STAT command.  Argument:
 | |
|         - message_spec: article number or message id (if not specified,
 | |
|           the current article is selected)
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - art_num: the article number
 | |
|         - message_id: the message id
 | |
|         """
 | |
|         if message_spec:
 | |
|             return self._statcmd('STAT {0}'.format(message_spec))
 | |
|         else:
 | |
|             return self._statcmd('STAT')
 | |
| 
 | |
|     def next(self):
 | |
|         """Process a NEXT command.  No arguments.  Return as for STAT."""
 | |
|         return self._statcmd('NEXT')
 | |
| 
 | |
|     def last(self):
 | |
|         """Process a LAST command.  No arguments.  Return as for STAT."""
 | |
|         return self._statcmd('LAST')
 | |
| 
 | |
|     def _artcmd(self, line, file=None):
 | |
|         """Internal: process a HEAD, BODY or ARTICLE command."""
 | |
|         resp, lines = self._longcmd(line, file)
 | |
|         resp, art_num, message_id = self._statparse(resp)
 | |
|         return resp, ArticleInfo(art_num, message_id, lines)
 | |
| 
 | |
|     def head(self, message_spec=None, *, file=None):
 | |
|         """Process a HEAD command.  Argument:
 | |
|         - message_spec: article number or message id
 | |
|         - file: filename string or file object to store the headers in
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - ArticleInfo: (article number, message id, list of header lines)
 | |
|         """
 | |
|         if message_spec is not None:
 | |
|             cmd = 'HEAD {0}'.format(message_spec)
 | |
|         else:
 | |
|             cmd = 'HEAD'
 | |
|         return self._artcmd(cmd, file)
 | |
| 
 | |
|     def body(self, message_spec=None, *, file=None):
 | |
|         """Process a BODY command.  Argument:
 | |
|         - message_spec: article number or message id
 | |
|         - file: filename string or file object to store the body in
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - ArticleInfo: (article number, message id, list of body lines)
 | |
|         """
 | |
|         if message_spec is not None:
 | |
|             cmd = 'BODY {0}'.format(message_spec)
 | |
|         else:
 | |
|             cmd = 'BODY'
 | |
|         return self._artcmd(cmd, file)
 | |
| 
 | |
|     def article(self, message_spec=None, *, file=None):
 | |
|         """Process an ARTICLE command.  Argument:
 | |
|         - message_spec: article number or message id
 | |
|         - file: filename string or file object to store the article in
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - ArticleInfo: (article number, message id, list of article lines)
 | |
|         """
 | |
|         if message_spec is not None:
 | |
|             cmd = 'ARTICLE {0}'.format(message_spec)
 | |
|         else:
 | |
|             cmd = 'ARTICLE'
 | |
|         return self._artcmd(cmd, file)
 | |
| 
 | |
|     def slave(self):
 | |
|         """Process a SLAVE command.  Returns:
 | |
|         - resp: server response if successful
 | |
|         """
 | |
|         return self._shortcmd('SLAVE')
 | |
| 
 | |
|     def xhdr(self, hdr, str, *, file=None):
 | |
|         """Process an XHDR command (optional server extension).  Arguments:
 | |
|         - hdr: the header type (e.g. 'subject')
 | |
|         - str: an article nr, a message id, or a range nr1-nr2
 | |
|         - file: Filename string or file object to store the result in
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - list: list of (nr, value) strings
 | |
|         """
 | |
|         pat = re.compile('^([0-9]+) ?(.*)\n?')
 | |
|         resp, lines = self._longcmdstring('XHDR {0} {1}'.format(hdr, str), file)
 | |
|         def remove_number(line):
 | |
|             m = pat.match(line)
 | |
|             return m.group(1, 2) if m else line
 | |
|         return resp, [remove_number(line) for line in lines]
 | |
| 
 | |
|     def xover(self, start, end, *, file=None):
 | |
|         """Process an XOVER command (optional server extension) Arguments:
 | |
|         - start: start of range
 | |
|         - end: end of range
 | |
|         - file: Filename string or file object to store the result in
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - list: list of dicts containing the response fields
 | |
|         """
 | |
|         resp, lines = self._longcmdstring('XOVER {0}-{1}'.format(start, end),
 | |
|                                           file)
 | |
|         fmt = self._getoverviewfmt()
 | |
|         return resp, _parse_overview(lines, fmt)
 | |
| 
 | |
|     def over(self, message_spec, *, file=None):
 | |
|         """Process an OVER command.  If the command isn't supported, fall
 | |
|         back to XOVER. Arguments:
 | |
|         - message_spec:
 | |
|             - either a message id, indicating the article to fetch
 | |
|               information about
 | |
|             - or a (start, end) tuple, indicating a range of article numbers;
 | |
|               if end is None, information up to the newest message will be
 | |
|               retrieved
 | |
|             - or None, indicating the current article number must be used
 | |
|         - file: Filename string or file object to store the result in
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - list: list of dicts containing the response fields
 | |
| 
 | |
|         NOTE: the "message id" form isn't supported by XOVER
 | |
|         """
 | |
|         cmd = 'OVER' if 'OVER' in self._caps else 'XOVER'
 | |
|         if isinstance(message_spec, (tuple, list)):
 | |
|             start, end = message_spec
 | |
|             cmd += ' {0}-{1}'.format(start, end or '')
 | |
|         elif message_spec is not None:
 | |
|             cmd = cmd + ' ' + message_spec
 | |
|         resp, lines = self._longcmdstring(cmd, file)
 | |
|         fmt = self._getoverviewfmt()
 | |
|         return resp, _parse_overview(lines, fmt)
 | |
| 
 | |
|     def xgtitle(self, group, *, file=None):
 | |
|         """Process an XGTITLE command (optional server extension) Arguments:
 | |
|         - group: group name wildcard (i.e. news.*)
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - list: list of (name,title) strings"""
 | |
|         warnings.warn("The XGTITLE extension is not actively used, "
 | |
|                       "use descriptions() instead",
 | |
|                       PendingDeprecationWarning, 2)
 | |
|         line_pat = re.compile('^([^ \t]+)[ \t]+(.*)$')
 | |
|         resp, raw_lines = self._longcmdstring('XGTITLE ' + group, file)
 | |
|         lines = []
 | |
|         for raw_line in raw_lines:
 | |
|             match = line_pat.search(raw_line.strip())
 | |
|             if match:
 | |
|                 lines.append(match.group(1, 2))
 | |
|         return resp, lines
 | |
| 
 | |
|     def xpath(self, id):
 | |
|         """Process an XPATH command (optional server extension) Arguments:
 | |
|         - id: Message id of article
 | |
|         Returns:
 | |
|         resp: server response if successful
 | |
|         path: directory path to article
 | |
|         """
 | |
|         warnings.warn("The XPATH extension is not actively used",
 | |
|                       PendingDeprecationWarning, 2)
 | |
| 
 | |
|         resp = self._shortcmd('XPATH {0}'.format(id))
 | |
|         if not resp.startswith('223'):
 | |
|             raise NNTPReplyError(resp)
 | |
|         try:
 | |
|             [resp_num, path] = resp.split()
 | |
|         except ValueError:
 | |
|             raise NNTPReplyError(resp)
 | |
|         else:
 | |
|             return resp, path
 | |
| 
 | |
|     def date(self):
 | |
|         """Process the DATE command.
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         - date: datetime object
 | |
|         """
 | |
|         resp = self._shortcmd("DATE")
 | |
|         if not resp.startswith('111'):
 | |
|             raise NNTPReplyError(resp)
 | |
|         elem = resp.split()
 | |
|         if len(elem) != 2:
 | |
|             raise NNTPDataError(resp)
 | |
|         date = elem[1]
 | |
|         if len(date) != 14:
 | |
|             raise NNTPDataError(resp)
 | |
|         return resp, _parse_datetime(date, None)
 | |
| 
 | |
|     def _post(self, command, f):
 | |
|         resp = self._shortcmd(command)
 | |
|         # Raises a specific exception if posting is not allowed
 | |
|         if not resp.startswith('3'):
 | |
|             raise NNTPReplyError(resp)
 | |
|         if isinstance(f, (bytes, bytearray)):
 | |
|             f = f.splitlines()
 | |
|         # We don't use _putline() because:
 | |
|         # - we don't want additional CRLF if the file or iterable is already
 | |
|         #   in the right format
 | |
|         # - we don't want a spurious flush() after each line is written
 | |
|         for line in f:
 | |
|             if not line.endswith(_CRLF):
 | |
|                 line = line.rstrip(b"\r\n") + _CRLF
 | |
|             if line.startswith(b'.'):
 | |
|                 line = b'.' + line
 | |
|             self.file.write(line)
 | |
|         self.file.write(b".\r\n")
 | |
|         self.file.flush()
 | |
|         return self._getresp()
 | |
| 
 | |
|     def post(self, data):
 | |
|         """Process a POST command.  Arguments:
 | |
|         - data: bytes object, iterable or file containing the article
 | |
|         Returns:
 | |
|         - resp: server response if successful"""
 | |
|         return self._post('POST', data)
 | |
| 
 | |
|     def ihave(self, message_id, data):
 | |
|         """Process an IHAVE command.  Arguments:
 | |
|         - message_id: message-id of the article
 | |
|         - data: file containing the article
 | |
|         Returns:
 | |
|         - resp: server response if successful
 | |
|         Note that if the server refuses the article an exception is raised."""
 | |
|         return self._post('IHAVE {0}'.format(message_id), data)
 | |
| 
 | |
|     def _close(self):
 | |
|         self.file.close()
 | |
|         del self.file
 | |
| 
 | |
|     def quit(self):
 | |
|         """Process a QUIT command and close the socket.  Returns:
 | |
|         - resp: server response if successful"""
 | |
|         try:
 | |
|             resp = self._shortcmd('QUIT')
 | |
|         finally:
 | |
|             self._close()
 | |
|         return resp
 | |
| 
 | |
|     def login(self, user=None, password=None, usenetrc=True):
 | |
|         if self.authenticated:
 | |
|             raise ValueError("Already logged in.")
 | |
|         if not user and not usenetrc:
 | |
|             raise ValueError(
 | |
|                 "At least one of `user` and `usenetrc` must be specified")
 | |
|         # If no login/password was specified but netrc was requested,
 | |
|         # try to get them from ~/.netrc
 | |
|         # Presume that if .netrc has an entry, NNRP authentication is required.
 | |
|         try:
 | |
|             if usenetrc and not user:
 | |
|                 import netrc
 | |
|                 credentials = netrc.netrc()
 | |
|                 auth = credentials.authenticators(self.host)
 | |
|                 if auth:
 | |
|                     user = auth[0]
 | |
|                     password = auth[2]
 | |
|         except IOError:
 | |
|             pass
 | |
|         # Perform NNTP authentication if needed.
 | |
|         if not user:
 | |
|             return
 | |
|         resp = self._shortcmd('authinfo user ' + user)
 | |
|         if resp.startswith('381'):
 | |
|             if not password:
 | |
|                 raise NNTPReplyError(resp)
 | |
|             else:
 | |
|                 resp = self._shortcmd('authinfo pass ' + password)
 | |
|                 if not resp.startswith('281'):
 | |
|                     raise NNTPPermanentError(resp)
 | |
|         # Attempt to send mode reader if it was requested after login.
 | |
|         if self.readermode_afterauth:
 | |
|             self._setreadermode()
 | |
| 
 | |
|     def _setreadermode(self):
 | |
|         try:
 | |
|             self.welcome = self._shortcmd('mode reader')
 | |
|         except NNTPPermanentError:
 | |
|             # Error 5xx, probably 'not implemented'
 | |
|             pass
 | |
|         except NNTPTemporaryError as e:
 | |
|             if e.response.startswith('480'):
 | |
|                 # Need authorization before 'mode reader'
 | |
|                 self.readermode_afterauth = True
 | |
|             else:
 | |
|                 raise
 | |
| 
 | |
|     if _have_ssl:
 | |
|         def starttls(self, context=None):
 | |
|             """Process a STARTTLS command. Arguments:
 | |
|             - context: SSL context to use for the encrypted connection
 | |
|             """
 | |
|             # Per RFC 4642, STARTTLS MUST NOT be sent after authentication or if
 | |
|             # a TLS session already exists.
 | |
|             if self.tls_on:
 | |
|                 raise ValueError("TLS is already enabled.")
 | |
|             if self.authenticated:
 | |
|                 raise ValueError("TLS cannot be started after authentication.")
 | |
|             resp = self._shortcmd('STARTTLS')
 | |
|             if resp.startswith('382'):
 | |
|                 self.file.close()
 | |
|                 self.sock = _encrypt_on(self.sock, context)
 | |
|                 self.file = self.sock.makefile("rwb")
 | |
|                 self.tls_on = True
 | |
|                 # Capabilities may change after TLS starts up, so ask for them
 | |
|                 # again.
 | |
|                 self._caps = None
 | |
|                 self.getcapabilities()
 | |
|             else:
 | |
|                 raise NNTPError("TLS failed to start.")
 | |
| 
 | |
| 
 | |
| class NNTP(_NNTPBase):
 | |
| 
 | |
|     def __init__(self, host, port=NNTP_PORT, user=None, password=None,
 | |
|                  readermode=None, usenetrc=False,
 | |
|                  timeout=_GLOBAL_DEFAULT_TIMEOUT):
 | |
|         """Initialize an instance.  Arguments:
 | |
|         - host: hostname to connect to
 | |
|         - port: port to connect to (default the standard NNTP port)
 | |
|         - user: username to authenticate with
 | |
|         - password: password to use with username
 | |
|         - readermode: if true, send 'mode reader' command after
 | |
|                       connecting.
 | |
|         - usenetrc: allow loading username and password from ~/.netrc file
 | |
|                     if not specified explicitly
 | |
|         - timeout: timeout (in seconds) used for socket connections
 | |
| 
 | |
|         readermode is sometimes necessary if you are connecting to an
 | |
|         NNTP server on the local machine and intend to call
 | |
|         reader-specific commands, such as `group'.  If you get
 | |
|         unexpected NNTPPermanentErrors, you might need to set
 | |
|         readermode.
 | |
|         """
 | |
|         self.host = host
 | |
|         self.port = port
 | |
|         self.sock = socket.create_connection((host, port), timeout)
 | |
|         file = self.sock.makefile("rwb")
 | |
|         _NNTPBase.__init__(self, file, host,
 | |
|                            readermode, timeout)
 | |
|         if user or usenetrc:
 | |
|             self.login(user, password, usenetrc)
 | |
| 
 | |
|     def _close(self):
 | |
|         try:
 | |
|             _NNTPBase._close(self)
 | |
|         finally:
 | |
|             self.sock.close()
 | |
| 
 | |
| 
 | |
| if _have_ssl:
 | |
|     class NNTP_SSL(_NNTPBase):
 | |
| 
 | |
|         def __init__(self, host, port=NNTP_SSL_PORT,
 | |
|                     user=None, password=None, ssl_context=None,
 | |
|                     readermode=None, usenetrc=False,
 | |
|                     timeout=_GLOBAL_DEFAULT_TIMEOUT):
 | |
|             """This works identically to NNTP.__init__, except for the change
 | |
|             in default port and the `ssl_context` argument for SSL connections.
 | |
|             """
 | |
|             self.sock = socket.create_connection((host, port), timeout)
 | |
|             self.sock = _encrypt_on(self.sock, ssl_context)
 | |
|             file = self.sock.makefile("rwb")
 | |
|             _NNTPBase.__init__(self, file, host,
 | |
|                                readermode=readermode, timeout=timeout)
 | |
|             if user or usenetrc:
 | |
|                 self.login(user, password, usenetrc)
 | |
| 
 | |
|         def _close(self):
 | |
|             try:
 | |
|                 _NNTPBase._close(self)
 | |
|             finally:
 | |
|                 self.sock.close()
 | |
| 
 | |
|     __all__.append("NNTP_SSL")
 | |
| 
 | |
| 
 | |
| # Test retrieval when run as a script.
 | |
| if __name__ == '__main__':
 | |
|     import argparse
 | |
|     from email.utils import parsedate
 | |
| 
 | |
|     parser = argparse.ArgumentParser(description="""\
 | |
|         nntplib built-in demo - display the latest articles in a newsgroup""")
 | |
|     parser.add_argument('-g', '--group', default='gmane.comp.python.general',
 | |
|                         help='group to fetch messages from (default: %(default)s)')
 | |
|     parser.add_argument('-s', '--server', default='news.gmane.org',
 | |
|                         help='NNTP server hostname (default: %(default)s)')
 | |
|     parser.add_argument('-p', '--port', default=-1, type=int,
 | |
|                         help='NNTP port number (default: %s / %s)' % (NNTP_PORT, NNTP_SSL_PORT))
 | |
|     parser.add_argument('-n', '--nb-articles', default=10, type=int,
 | |
|                         help='number of articles to fetch (default: %(default)s)')
 | |
|     parser.add_argument('-S', '--ssl', action='store_true', default=False,
 | |
|                         help='use NNTP over SSL')
 | |
|     args = parser.parse_args()
 | |
| 
 | |
|     port = args.port
 | |
|     if not args.ssl:
 | |
|         if port == -1:
 | |
|             port = NNTP_PORT
 | |
|         s = NNTP(host=args.server, port=port)
 | |
|     else:
 | |
|         if port == -1:
 | |
|             port = NNTP_SSL_PORT
 | |
|         s = NNTP_SSL(host=args.server, port=port)
 | |
| 
 | |
|     caps = s.getcapabilities()
 | |
|     if 'STARTTLS' in caps:
 | |
|         s.starttls()
 | |
|     resp, count, first, last, name = s.group(args.group)
 | |
|     print('Group', name, 'has', count, 'articles, range', first, 'to', last)
 | |
| 
 | |
|     def cut(s, lim):
 | |
|         if len(s) > lim:
 | |
|             s = s[:lim - 4] + "..."
 | |
|         return s
 | |
| 
 | |
|     first = str(int(last) - args.nb_articles + 1)
 | |
|     resp, overviews = s.xover(first, last)
 | |
|     for artnum, over in overviews:
 | |
|         author = decode_header(over['from']).split('<', 1)[0]
 | |
|         subject = decode_header(over['subject'])
 | |
|         lines = int(over[':lines'])
 | |
|         print("{:7} {:20} {:42} ({})".format(
 | |
|               artnum, cut(author, 20), cut(subject, 42), lines)
 | |
|               )
 | |
| 
 | |
|     s.quit()
 | 
