mirror of
				https://github.com/python/cpython.git
				synced 2025-10-30 21:21:22 +00:00 
			
		
		
		
	 806c2469cb
			
		
	
	
		806c2469cb
		
	
	
	
	
		
			
			svn+ssh://pythondev@svn.python.org/python/branches/p3yk
................
  r56760 | neal.norwitz | 2007-08-05 18:55:39 -0700 (Sun, 05 Aug 2007) | 178 lines
  Merged revisions 56477-56759 via svnmerge from
  svn+ssh://pythondev@svn.python.org/python/trunk
  ........
    r56485 | facundo.batista | 2007-07-21 17:13:00 -0700 (Sat, 21 Jul 2007) | 5 lines
    Selectively enable tests for asyncore.readwrite based on the presence
    of poll support in the select module (since this is the only case in
    which readwrite can be called). [GSoC - Alan McIntyre]
  ........
    r56488 | nick.coghlan | 2007-07-22 03:18:07 -0700 (Sun, 22 Jul 2007) | 1 line
    Add explicit relative import tests for runpy.run_module
  ........
    r56509 | nick.coghlan | 2007-07-23 06:41:45 -0700 (Mon, 23 Jul 2007) | 5 lines
    Correctly cleanup sys.modules after executing runpy relative import
    tests
    Restore Python 2.4 ImportError when attempting to execute a package
    (as imports cannot be guaranteed to work properly if you try it)
  ........
    r56519 | nick.coghlan | 2007-07-24 06:07:38 -0700 (Tue, 24 Jul 2007) | 1 line
    Tweak runpy test to do a better job of confirming that sys has been manipulated correctly
  ........
    r56520 | nick.coghlan | 2007-07-24 06:58:28 -0700 (Tue, 24 Jul 2007) | 1 line
    Fix an incompatibility between the -i and -m command line switches as reported on python-dev by PJE - runpy.run_module now leaves any changes it makes to the sys module intact after the function terminates
  ........
    r56523 | nick.coghlan | 2007-07-24 07:39:23 -0700 (Tue, 24 Jul 2007) | 1 line
    Try to get rid of spurious failure in test_resource on the Debian buildbots by changing the file size limit before attempting to close the file
  ........
    r56533 | facundo.batista | 2007-07-24 14:20:42 -0700 (Tue, 24 Jul 2007) | 7 lines
    New tests for basic behavior of smtplib.SMTP and
    smtpd.DebuggingServer. Change to use global host & port number
    variables. Modified the 'server' to take a string to send back in
    order to vary test server responses. Added a test for the reaction of
    smtplib.SMTP to a non-200 HELO response. [GSoC - Alan McIntyre]
  ........
    r56538 | nick.coghlan | 2007-07-25 05:57:48 -0700 (Wed, 25 Jul 2007) | 1 line
    More buildbot cleanup - let the OS assign the port for test_urllib2_localnet
  ........
    r56539 | nick.coghlan | 2007-07-25 06:18:58 -0700 (Wed, 25 Jul 2007) | 1 line
    Add a temporary diagnostic message before a strange failure on the alpha Debian buildbot
  ........
    r56543 | martin.v.loewis | 2007-07-25 09:24:23 -0700 (Wed, 25 Jul 2007) | 2 lines
    Change location of the package index to pypi.python.org/pypi
  ........
    r56551 | georg.brandl | 2007-07-26 02:36:25 -0700 (Thu, 26 Jul 2007) | 2 lines
    tabs, newlines and crs are valid XML characters.
  ........
    r56553 | nick.coghlan | 2007-07-26 07:03:00 -0700 (Thu, 26 Jul 2007) | 1 line
    Add explicit test for a misbehaving math.floor
  ........
    r56561 | mark.hammond | 2007-07-26 21:52:32 -0700 (Thu, 26 Jul 2007) | 3 lines
    In consultation with Kristjan Jonsson, only define WINVER and _WINNT_WIN32
    if (a) we are building Python itself and (b) no one previously defined them
  ........
    r56562 | mark.hammond | 2007-07-26 22:08:54 -0700 (Thu, 26 Jul 2007) | 2 lines
    Correctly detect AMD64 architecture on VC2003
  ........
    r56566 | nick.coghlan | 2007-07-27 03:36:30 -0700 (Fri, 27 Jul 2007) | 1 line
    Make test_math error messages more meaningful for small discrepancies in results
  ........
    r56588 | martin.v.loewis | 2007-07-27 11:28:22 -0700 (Fri, 27 Jul 2007) | 2 lines
    Bug #978833: Close https sockets by releasing the _ssl object.
  ........
    r56601 | martin.v.loewis | 2007-07-28 00:03:05 -0700 (Sat, 28 Jul 2007) | 3 lines
    Bug #1704793: Return UTF-16 pair if unicodedata.lookup cannot
    represent the result in a single character.
  ........
    r56604 | facundo.batista | 2007-07-28 07:21:22 -0700 (Sat, 28 Jul 2007) | 9 lines
    Moved all of the capture_server socket setup code into the try block
    so that the event gets set if a failure occurs during server setup
    (otherwise the test will block forever).  Changed to let the OS assign
    the server port number, and client side of test waits for port number
    assignment before proceeding. The test data in DispatcherWithSendTests
    is also sent in multiple send() calls instead of one to make sure this
    works properly. [GSoC - Alan McIntyre]
  ........
    r56611 | georg.brandl | 2007-07-29 01:26:10 -0700 (Sun, 29 Jul 2007) | 2 lines
    Clarify PEP 343 description.
  ........
    r56614 | georg.brandl | 2007-07-29 02:11:15 -0700 (Sun, 29 Jul 2007) | 2 lines
    try-except-finally is new in 2.5.
  ........
    r56617 | facundo.batista | 2007-07-29 07:23:08 -0700 (Sun, 29 Jul 2007) | 9 lines
    Added tests for asynchat classes simple_producer & fifo, and the
    find_prefix_at_end function. Check behavior of a string given as a
    producer.  Added tests for behavior of asynchat.async_chat when given
    int, long, and None terminator arguments. Added usepoll attribute to
    TestAsynchat to allow running the asynchat tests with poll support
    chosen whether it's available or not (improves coverage of asyncore
    code). [GSoC - Alan McIntyre]
  ........
    r56620 | georg.brandl | 2007-07-29 10:38:35 -0700 (Sun, 29 Jul 2007) | 2 lines
    Bug #1763149: use proper slice syntax in docstring.
     (backport)
  ........
    r56624 | mark.hammond | 2007-07-29 17:45:29 -0700 (Sun, 29 Jul 2007) | 4 lines
    Correct use of Py_BUILD_CORE - now make sure it is defined before it is
    referenced, and also fix definition of _WIN32_WINNT.
    Resolves patch 1761803.
  ........
    r56632 | facundo.batista | 2007-07-30 20:03:34 -0700 (Mon, 30 Jul 2007) | 8 lines
    When running asynchat tests on OS X (darwin), the test client now
    overrides asyncore.dispatcher.handle_expt to do nothing, since
    select.poll gives a POLLHUP error at the completion of these tests.
    Added timeout & count arguments to several asyncore.loop calls to
    avoid the possibility of a test hanging up a build. [GSoC - Alan
    McIntyre]
  ........
    r56633 | nick.coghlan | 2007-07-31 06:38:01 -0700 (Tue, 31 Jul 2007) | 1 line
    Eliminate RLock race condition reported in SF bug #1764059
  ........
    r56636 | martin.v.loewis | 2007-07-31 12:57:56 -0700 (Tue, 31 Jul 2007) | 2 lines
    Define _BSD_SOURCE, to get access to POSIX extensions on OpenBSD 4.1+.
  ........
    r56653 | facundo.batista | 2007-08-01 16:18:36 -0700 (Wed, 01 Aug 2007) | 9 lines
    Allow the OS to select a free port for each test server. For
    DebuggingServerTests, construct SMTP objects with a localhost argument
    to avoid abysmally long FQDN lookups (not relevant to items under
    test) on some machines that would cause the test to fail. Moved server
    setup code in the server function inside the try block to avoid the
    possibility of setup failure hanging the test.  Minor edits to conform
    to PEP 8. [GSoC - Alan McIntyre]
  ........
    r56681 | matthias.klose | 2007-08-02 14:33:13 -0700 (Thu, 02 Aug 2007) | 2 lines
    - Allow Emacs 22 for building the documentation in info format.
  ........
    r56689 | neal.norwitz | 2007-08-02 23:46:29 -0700 (Thu, 02 Aug 2007) | 1 line
    Py_ssize_t is defined regardless of HAVE_LONG_LONG.  Will backport
  ........
    r56727 | hyeshik.chang | 2007-08-03 21:10:18 -0700 (Fri, 03 Aug 2007) | 3 lines
    Fix gb18030 codec's bug that doesn't map two-byte characters on
    GB18030 extension in encoding. (bug reported by Bjorn Stabell)
  ........
    r56751 | neal.norwitz | 2007-08-04 20:23:31 -0700 (Sat, 04 Aug 2007) | 7 lines
    Handle errors when generating a warning.
    The value is always written to the returned pointer if getting it was
    successful, even if a warning causes an error. (This probably doesn't matter
    as the caller will probably discard the value.)
    Will backport.
  ........
................
		
	
			
		
			
				
	
	
		
			299 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			299 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """distutils.command.register
 | |
| 
 | |
| Implements the Distutils 'register' command (register with the repository).
 | |
| """
 | |
| 
 | |
| # created 2002/10/21, Richard Jones
 | |
| 
 | |
| __revision__ = "$Id$"
 | |
| 
 | |
| import sys, os, urllib2, getpass, urlparse
 | |
| import StringIO, ConfigParser
 | |
| 
 | |
| from distutils.core import Command
 | |
| from distutils.errors import *
 | |
| 
 | |
| def raw_input(prompt):
 | |
|     sys.stdout.write(prompt)
 | |
|     sys.stdout.flush()
 | |
|     return sys.stdin.readline()
 | |
| 
 | |
| class register(Command):
 | |
| 
 | |
|     description = ("register the distribution with the Python package index")
 | |
| 
 | |
|     DEFAULT_REPOSITORY = 'http://pypi.python.org/pypi'
 | |
| 
 | |
|     user_options = [
 | |
|         ('repository=', 'r',
 | |
|          "url of repository [default: %s]"%DEFAULT_REPOSITORY),
 | |
|         ('list-classifiers', None,
 | |
|          'list the valid Trove classifiers'),
 | |
|         ('show-response', None,
 | |
|          'display full response text from server'),
 | |
|         ]
 | |
|     boolean_options = ['verify', 'show-response', 'list-classifiers']
 | |
| 
 | |
|     def initialize_options(self):
 | |
|         self.repository = None
 | |
|         self.show_response = 0
 | |
|         self.list_classifiers = 0
 | |
| 
 | |
|     def finalize_options(self):
 | |
|         if self.repository is None:
 | |
|             self.repository = self.DEFAULT_REPOSITORY
 | |
| 
 | |
|     def run(self):
 | |
|         self.check_metadata()
 | |
|         if self.dry_run:
 | |
|             self.verify_metadata()
 | |
|         elif self.list_classifiers:
 | |
|             self.classifiers()
 | |
|         else:
 | |
|             self.send_metadata()
 | |
| 
 | |
|     def check_metadata(self):
 | |
|         """Ensure that all required elements of meta-data (name, version,
 | |
|            URL, (author and author_email) or (maintainer and
 | |
|            maintainer_email)) are supplied by the Distribution object; warn if
 | |
|            any are missing.
 | |
|         """
 | |
|         metadata = self.distribution.metadata
 | |
| 
 | |
|         missing = []
 | |
|         for attr in ('name', 'version', 'url'):
 | |
|             if not (hasattr(metadata, attr) and getattr(metadata, attr)):
 | |
|                 missing.append(attr)
 | |
| 
 | |
|         if missing:
 | |
|             self.warn("missing required meta-data: " +
 | |
|                       ", ".join(missing))
 | |
| 
 | |
|         if metadata.author:
 | |
|             if not metadata.author_email:
 | |
|                 self.warn("missing meta-data: if 'author' supplied, " +
 | |
|                           "'author_email' must be supplied too")
 | |
|         elif metadata.maintainer:
 | |
|             if not metadata.maintainer_email:
 | |
|                 self.warn("missing meta-data: if 'maintainer' supplied, " +
 | |
|                           "'maintainer_email' must be supplied too")
 | |
|         else:
 | |
|             self.warn("missing meta-data: either (author and author_email) " +
 | |
|                       "or (maintainer and maintainer_email) " +
 | |
|                       "must be supplied")
 | |
| 
 | |
|     def classifiers(self):
 | |
|         ''' Fetch the list of classifiers from the server.
 | |
|         '''
 | |
|         response = urllib2.urlopen(self.repository+'?:action=list_classifiers')
 | |
|         print(response.read())
 | |
| 
 | |
|     def verify_metadata(self):
 | |
|         ''' Send the metadata to the package index server to be checked.
 | |
|         '''
 | |
|         # send the info to the server and report the result
 | |
|         (code, result) = self.post_to_server(self.build_post_data('verify'))
 | |
|         print('Server response (%s): %s'%(code, result))
 | |
| 
 | |
|     def send_metadata(self):
 | |
|         ''' Send the metadata to the package index server.
 | |
| 
 | |
|             Well, do the following:
 | |
|             1. figure who the user is, and then
 | |
|             2. send the data as a Basic auth'ed POST.
 | |
| 
 | |
|             First we try to read the username/password from $HOME/.pypirc,
 | |
|             which is a ConfigParser-formatted file with a section
 | |
|             [server-login] containing username and password entries (both
 | |
|             in clear text). Eg:
 | |
| 
 | |
|                 [server-login]
 | |
|                 username: fred
 | |
|                 password: sekrit
 | |
| 
 | |
|             Otherwise, to figure who the user is, we offer the user three
 | |
|             choices:
 | |
| 
 | |
|              1. use existing login,
 | |
|              2. register as a new user, or
 | |
|              3. set the password to a random string and email the user.
 | |
| 
 | |
|         '''
 | |
|         choice = 'x'
 | |
|         username = password = ''
 | |
| 
 | |
|         # see if we can short-cut and get the username/password from the
 | |
|         # config
 | |
|         config = None
 | |
|         if os.environ.has_key('HOME'):
 | |
|             rc = os.path.join(os.environ['HOME'], '.pypirc')
 | |
|             if os.path.exists(rc):
 | |
|                 print('Using PyPI login from %s'%rc)
 | |
|                 config = ConfigParser.ConfigParser()
 | |
|                 config.read(rc)
 | |
|                 username = config.get('server-login', 'username')
 | |
|                 password = config.get('server-login', 'password')
 | |
|                 choice = '1'
 | |
| 
 | |
|         # get the user's login info
 | |
|         choices = '1 2 3 4'.split()
 | |
|         while choice not in choices:
 | |
|             print('''We need to know who you are, so please choose either:
 | |
|  1. use your existing login,
 | |
|  2. register as a new user,
 | |
|  3. have the server generate a new password for you (and email it to you), or
 | |
|  4. quit
 | |
| Your selection [default 1]: ''', end=' ')
 | |
|             choice = raw_input()
 | |
|             if not choice:
 | |
|                 choice = '1'
 | |
|             elif choice not in choices:
 | |
|                 print('Please choose one of the four options!')
 | |
| 
 | |
|         if choice == '1':
 | |
|             # get the username and password
 | |
|             while not username:
 | |
|                 username = raw_input('Username: ')
 | |
|             while not password:
 | |
|                 password = getpass.getpass('Password: ')
 | |
| 
 | |
|             # set up the authentication
 | |
|             auth = urllib2.HTTPPasswordMgr()
 | |
|             host = urlparse.urlparse(self.repository)[1]
 | |
|             auth.add_password('pypi', host, username, password)
 | |
| 
 | |
|             # send the info to the server and report the result
 | |
|             code, result = self.post_to_server(self.build_post_data('submit'),
 | |
|                 auth)
 | |
|             print('Server response (%s): %s'%(code, result))
 | |
| 
 | |
|             # possibly save the login
 | |
|             if os.environ.has_key('HOME') and config is None and code == 200:
 | |
|                 rc = os.path.join(os.environ['HOME'], '.pypirc')
 | |
|                 print('I can store your PyPI login so future submissions will be faster.')
 | |
|                 print('(the login will be stored in %s)'%rc)
 | |
|                 choice = 'X'
 | |
|                 while choice.lower() not in 'yn':
 | |
|                     choice = raw_input('Save your login (y/N)?')
 | |
|                     if not choice:
 | |
|                         choice = 'n'
 | |
|                 if choice.lower() == 'y':
 | |
|                     f = open(rc, 'w')
 | |
|                     f.write('[server-login]\nusername:%s\npassword:%s\n'%(
 | |
|                         username, password))
 | |
|                     f.close()
 | |
|                     try:
 | |
|                         os.chmod(rc, 0o600)
 | |
|                     except:
 | |
|                         pass
 | |
|         elif choice == '2':
 | |
|             data = {':action': 'user'}
 | |
|             data['name'] = data['password'] = data['email'] = ''
 | |
|             data['confirm'] = None
 | |
|             while not data['name']:
 | |
|                 data['name'] = raw_input('Username: ')
 | |
|             while data['password'] != data['confirm']:
 | |
|                 while not data['password']:
 | |
|                     data['password'] = getpass.getpass('Password: ')
 | |
|                 while not data['confirm']:
 | |
|                     data['confirm'] = getpass.getpass(' Confirm: ')
 | |
|                 if data['password'] != data['confirm']:
 | |
|                     data['password'] = ''
 | |
|                     data['confirm'] = None
 | |
|                     print("Password and confirm don't match!")
 | |
|             while not data['email']:
 | |
|                 data['email'] = raw_input('   EMail: ')
 | |
|             code, result = self.post_to_server(data)
 | |
|             if code != 200:
 | |
|                 print('Server response (%s): %s'%(code, result))
 | |
|             else:
 | |
|                 print('You will receive an email shortly.')
 | |
|                 print('Follow the instructions in it to complete registration.')
 | |
|         elif choice == '3':
 | |
|             data = {':action': 'password_reset'}
 | |
|             data['email'] = ''
 | |
|             while not data['email']:
 | |
|                 data['email'] = raw_input('Your email address: ')
 | |
|             code, result = self.post_to_server(data)
 | |
|             print('Server response (%s): %s'%(code, result))
 | |
| 
 | |
|     def build_post_data(self, action):
 | |
|         # figure the data to send - the metadata plus some additional
 | |
|         # information used by the package server
 | |
|         meta = self.distribution.metadata
 | |
|         data = {
 | |
|             ':action': action,
 | |
|             'metadata_version' : '1.0',
 | |
|             'name': meta.get_name(),
 | |
|             'version': meta.get_version(),
 | |
|             'summary': meta.get_description(),
 | |
|             'home_page': meta.get_url(),
 | |
|             'author': meta.get_contact(),
 | |
|             'author_email': meta.get_contact_email(),
 | |
|             'license': meta.get_licence(),
 | |
|             'description': meta.get_long_description(),
 | |
|             'keywords': meta.get_keywords(),
 | |
|             'platform': meta.get_platforms(),
 | |
|             'classifiers': meta.get_classifiers(),
 | |
|             'download_url': meta.get_download_url(),
 | |
|             # PEP 314
 | |
|             'provides': meta.get_provides(),
 | |
|             'requires': meta.get_requires(),
 | |
|             'obsoletes': meta.get_obsoletes(),
 | |
|         }
 | |
|         if data['provides'] or data['requires'] or data['obsoletes']:
 | |
|             data['metadata_version'] = '1.1'
 | |
|         return data
 | |
| 
 | |
|     def post_to_server(self, data, auth=None):
 | |
|         ''' Post a query to the server, and return a string response.
 | |
|         '''
 | |
| 
 | |
|         # Build up the MIME payload for the urllib2 POST data
 | |
|         boundary = '--------------GHSKFJDLGDS7543FJKLFHRE75642756743254'
 | |
|         sep_boundary = '\n--' + boundary
 | |
|         end_boundary = sep_boundary + '--'
 | |
|         body = StringIO.StringIO()
 | |
|         for key, value in data.items():
 | |
|             # handle multiple entries for the same name
 | |
|             if type(value) not in (type([]), type( () )):
 | |
|                 value = [value]
 | |
|             for value in value:
 | |
|                 value = str(value).encode("utf-8")
 | |
|                 body.write(sep_boundary)
 | |
|                 body.write('\nContent-Disposition: form-data; name="%s"'%key)
 | |
|                 body.write("\n\n")
 | |
|                 body.write(value)
 | |
|                 if value and value[-1] == '\r':
 | |
|                     body.write('\n')  # write an extra newline (lurve Macs)
 | |
|         body.write(end_boundary)
 | |
|         body.write("\n")
 | |
|         body = body.getvalue()
 | |
| 
 | |
|         # build the Request
 | |
|         headers = {
 | |
|             'Content-type': 'multipart/form-data; boundary=%s; charset=utf-8'%boundary,
 | |
|             'Content-length': str(len(body))
 | |
|         }
 | |
|         req = urllib2.Request(self.repository, body, headers)
 | |
| 
 | |
|         # handle HTTP and include the Basic Auth handler
 | |
|         opener = urllib2.build_opener(
 | |
|             urllib2.HTTPBasicAuthHandler(password_mgr=auth)
 | |
|         )
 | |
|         data = ''
 | |
|         try:
 | |
|             result = opener.open(req)
 | |
|         except urllib2.HTTPError as e:
 | |
|             if self.show_response:
 | |
|                 data = e.fp.read()
 | |
|             result = e.code, e.msg
 | |
|         except urllib2.URLError as e:
 | |
|             result = 500, str(e)
 | |
|         else:
 | |
|             if self.show_response:
 | |
|                 data = result.read()
 | |
|             result = 200, 'OK'
 | |
|         if self.show_response:
 | |
|             print('-'*75, data, '-'*75)
 | |
|         return result
 |