2009-09-09 14:56:53 -05:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
"""
|
|
|
|
test_intersphinx
|
|
|
|
~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Test the intersphinx extension.
|
|
|
|
|
2016-01-14 15:54:04 -06:00
|
|
|
:copyright: Copyright 2007-2016 by the Sphinx team, see AUTHORS.
|
2009-09-09 14:56:53 -05:00
|
|
|
:license: BSD, see LICENSE for details.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import posixpath
|
2015-10-15 16:37:55 -05:00
|
|
|
import unittest
|
|
|
|
import zlib
|
2009-09-09 14:56:53 -05:00
|
|
|
|
2014-04-30 07:30:46 -05:00
|
|
|
from six import BytesIO
|
2009-09-09 14:56:53 -05:00
|
|
|
from docutils import nodes
|
|
|
|
|
|
|
|
from sphinx import addnodes
|
2016-08-17 10:56:40 -05:00
|
|
|
from sphinx.ext.intersphinx import read_inventory, \
|
2015-10-15 16:37:55 -05:00
|
|
|
load_mappings, missing_reference, _strip_basic_auth, _read_from_url, \
|
2016-08-10 21:19:42 -05:00
|
|
|
_get_safe_url, fetch_inventory, INVENTORY_FILENAME
|
2009-09-09 14:56:53 -05:00
|
|
|
|
2015-10-15 16:37:55 -05:00
|
|
|
from util import with_app, with_tempdir, mock
|
2009-09-09 14:56:53 -05:00
|
|
|
|
|
|
|
|
|
|
|
inventory_v1 = '''\
|
|
|
|
# Sphinx inventory version 1
|
|
|
|
# Project: foo
|
|
|
|
# Version: 1.0
|
|
|
|
module mod foo.html
|
|
|
|
module.cls class foo.html
|
2010-06-06 18:34:01 -05:00
|
|
|
'''.encode('utf-8')
|
2009-09-09 14:56:53 -05:00
|
|
|
|
|
|
|
inventory_v2 = '''\
|
|
|
|
# Sphinx inventory version 2
|
|
|
|
# Project: foo
|
|
|
|
# Version: 2.0
|
|
|
|
# The remainder of this file is compressed with zlib.
|
2010-06-06 18:34:01 -05:00
|
|
|
'''.encode('utf-8') + zlib.compress('''\
|
2010-05-23 06:06:01 -05:00
|
|
|
module1 py:module 0 foo.html#module-module1 Long Module desc
|
|
|
|
module2 py:module 0 foo.html#module-$ -
|
|
|
|
module1.func py:function 1 sub/foo.html#$ -
|
|
|
|
CFunc c:function 2 cfunc.html#CFunc -
|
2012-02-26 09:49:07 -06:00
|
|
|
a term std:term -1 glossary.html#term-a-term -
|
2016-06-24 07:01:35 -05:00
|
|
|
a term including:colon std:term -1 glossary.html#term-a-term-including-colon -
|
2010-06-06 18:34:01 -05:00
|
|
|
'''.encode('utf-8'))
|
2009-09-09 14:56:53 -05:00
|
|
|
|
|
|
|
|
|
|
|
def test_read_inventory_v1():
|
2014-04-30 07:30:46 -05:00
|
|
|
f = BytesIO(inventory_v1)
|
2016-08-17 10:56:40 -05:00
|
|
|
invdata = read_inventory(f, '/util', posixpath.join)
|
2009-09-09 14:56:53 -05:00
|
|
|
assert invdata['py:module']['module'] == \
|
2014-09-21 10:17:02 -05:00
|
|
|
('foo', '1.0', '/util/foo.html#module-module', '-')
|
2009-09-09 14:56:53 -05:00
|
|
|
assert invdata['py:class']['module.cls'] == \
|
2014-09-21 10:17:02 -05:00
|
|
|
('foo', '1.0', '/util/foo.html#module.cls', '-')
|
2009-09-09 14:56:53 -05:00
|
|
|
|
|
|
|
|
|
|
|
def test_read_inventory_v2():
|
2014-04-30 07:30:46 -05:00
|
|
|
f = BytesIO(inventory_v2)
|
2016-08-17 10:56:40 -05:00
|
|
|
invdata1 = read_inventory(f, '/util', posixpath.join)
|
2009-09-09 14:56:53 -05:00
|
|
|
|
|
|
|
# try again with a small buffer size to test the chunking algorithm
|
2014-04-30 07:30:46 -05:00
|
|
|
f = BytesIO(inventory_v2)
|
2016-08-17 10:56:40 -05:00
|
|
|
invdata2 = read_inventory(f, '/util', posixpath.join, bufsize=5)
|
2009-09-09 14:56:53 -05:00
|
|
|
|
|
|
|
assert invdata1 == invdata2
|
|
|
|
|
|
|
|
assert len(invdata1['py:module']) == 2
|
|
|
|
assert invdata1['py:module']['module1'] == \
|
2014-09-21 10:17:02 -05:00
|
|
|
('foo', '2.0', '/util/foo.html#module-module1', 'Long Module desc')
|
2009-09-09 14:56:53 -05:00
|
|
|
assert invdata1['py:module']['module2'] == \
|
2014-09-21 10:17:02 -05:00
|
|
|
('foo', '2.0', '/util/foo.html#module-module2', '-')
|
2009-09-09 14:56:53 -05:00
|
|
|
assert invdata1['py:function']['module1.func'][2] == \
|
2014-09-21 10:17:02 -05:00
|
|
|
'/util/sub/foo.html#module1.func'
|
2009-09-09 14:56:53 -05:00
|
|
|
assert invdata1['c:function']['CFunc'][2] == '/util/cfunc.html#CFunc'
|
2012-02-26 09:49:07 -06:00
|
|
|
assert invdata1['std:term']['a term'][2] == \
|
2014-09-21 10:17:02 -05:00
|
|
|
'/util/glossary.html#term-a-term'
|
2016-06-24 07:01:35 -05:00
|
|
|
assert invdata1['std:term']['a term including:colon'][2] == \
|
|
|
|
'/util/glossary.html#term-a-term-including-colon'
|
2009-09-09 14:56:53 -05:00
|
|
|
|
|
|
|
|
2016-08-10 21:19:42 -05:00
|
|
|
@with_app()
|
2016-08-17 22:33:28 -05:00
|
|
|
@mock.patch('sphinx.ext.intersphinx.read_inventory')
|
2016-08-10 21:19:42 -05:00
|
|
|
@mock.patch('sphinx.ext.intersphinx._read_from_url')
|
2016-08-17 22:33:28 -05:00
|
|
|
def test_fetch_inventory_redirection(app, status, warning, _read_from_url, read_inventory):
|
2016-08-10 23:58:21 -05:00
|
|
|
_read_from_url().readline.return_value = '# Sphinx inventory version 2'.encode('utf-8')
|
2016-08-10 21:19:42 -05:00
|
|
|
|
|
|
|
# same uri and inv, not redirected
|
2016-08-17 22:33:28 -05:00
|
|
|
_read_from_url().url = 'http://hostname/' + INVENTORY_FILENAME
|
2016-08-10 21:19:42 -05:00
|
|
|
fetch_inventory(app, 'http://hostname/', 'http://hostname/' + INVENTORY_FILENAME)
|
|
|
|
assert 'intersphinx inventory has moved' not in status.getvalue()
|
2016-08-17 22:33:28 -05:00
|
|
|
assert read_inventory.call_args[0][1] == 'http://hostname/'
|
2016-08-10 21:19:42 -05:00
|
|
|
|
|
|
|
# same uri and inv, redirected
|
2016-08-10 23:58:21 -05:00
|
|
|
status.seek(0)
|
2016-08-10 21:19:42 -05:00
|
|
|
status.truncate(0)
|
2016-08-17 22:33:28 -05:00
|
|
|
_read_from_url().url = 'http://hostname/new/' + INVENTORY_FILENAME
|
2016-08-10 21:19:42 -05:00
|
|
|
|
|
|
|
fetch_inventory(app, 'http://hostname/', 'http://hostname/' + INVENTORY_FILENAME)
|
|
|
|
assert status.getvalue() == ('intersphinx inventory has moved: '
|
|
|
|
'http://hostname/%s -> http://hostname/new/%s\n' %
|
|
|
|
(INVENTORY_FILENAME, INVENTORY_FILENAME))
|
2016-08-17 22:33:28 -05:00
|
|
|
assert read_inventory.call_args[0][1] == 'http://hostname/new'
|
2016-08-10 21:19:42 -05:00
|
|
|
|
|
|
|
# different uri and inv, not redirected
|
2016-08-10 23:58:21 -05:00
|
|
|
status.seek(0)
|
2016-08-10 21:19:42 -05:00
|
|
|
status.truncate(0)
|
2016-08-17 22:33:28 -05:00
|
|
|
_read_from_url().url = 'http://hostname/new/' + INVENTORY_FILENAME
|
2016-08-10 21:19:42 -05:00
|
|
|
|
|
|
|
fetch_inventory(app, 'http://hostname/', 'http://hostname/new/' + INVENTORY_FILENAME)
|
|
|
|
assert 'intersphinx inventory has moved' not in status.getvalue()
|
2016-08-17 22:33:28 -05:00
|
|
|
assert read_inventory.call_args[0][1] == 'http://hostname/'
|
2016-08-10 21:19:42 -05:00
|
|
|
|
|
|
|
# different uri and inv, redirected
|
2016-08-10 23:58:21 -05:00
|
|
|
status.seek(0)
|
2016-08-10 21:19:42 -05:00
|
|
|
status.truncate(0)
|
2016-08-17 22:33:28 -05:00
|
|
|
_read_from_url().url = 'http://hostname/other/' + INVENTORY_FILENAME
|
2016-08-10 21:19:42 -05:00
|
|
|
|
|
|
|
fetch_inventory(app, 'http://hostname/', 'http://hostname/new/' + INVENTORY_FILENAME)
|
|
|
|
assert status.getvalue() == ('intersphinx inventory has moved: '
|
|
|
|
'http://hostname/new/%s -> http://hostname/other/%s\n' %
|
|
|
|
(INVENTORY_FILENAME, INVENTORY_FILENAME))
|
2016-08-17 22:33:28 -05:00
|
|
|
assert read_inventory.call_args[0][1] == 'http://hostname/'
|
2016-08-10 21:19:42 -05:00
|
|
|
|
|
|
|
|
2014-01-10 15:10:29 -06:00
|
|
|
@with_app()
|
2009-09-09 14:56:53 -05:00
|
|
|
@with_tempdir
|
2014-09-21 10:17:02 -05:00
|
|
|
def test_missing_reference(tempdir, app, status, warning):
|
2009-09-09 14:56:53 -05:00
|
|
|
inv_file = tempdir / 'inventory'
|
2014-04-28 05:58:26 -05:00
|
|
|
inv_file.write_bytes(inventory_v2)
|
2010-07-27 06:20:58 -05:00
|
|
|
app.config.intersphinx_mapping = {
|
2015-02-22 22:20:35 -06:00
|
|
|
'https://docs.python.org/': inv_file,
|
|
|
|
'py3k': ('https://docs.python.org/py3k/', inv_file),
|
2010-07-27 06:20:58 -05:00
|
|
|
}
|
2009-09-09 14:56:53 -05:00
|
|
|
app.config.intersphinx_cache_limit = 0
|
|
|
|
|
|
|
|
# load the inventory and check if it's done correctly
|
|
|
|
load_mappings(app)
|
|
|
|
inv = app.env.intersphinx_inventory
|
|
|
|
|
|
|
|
assert inv['py:module']['module2'] == \
|
2015-02-22 22:20:35 -06:00
|
|
|
('foo', '2.0', 'https://docs.python.org/foo.html#module-module2', '-')
|
2009-09-09 14:56:53 -05:00
|
|
|
|
|
|
|
# create fake nodes and check referencing
|
|
|
|
|
2010-08-05 04:58:43 -05:00
|
|
|
def fake_node(domain, type, target, content, **attrs):
|
|
|
|
contnode = nodes.emphasis(content, content)
|
|
|
|
node = addnodes.pending_xref('')
|
|
|
|
node['reftarget'] = target
|
|
|
|
node['reftype'] = type
|
|
|
|
node['refdomain'] = domain
|
|
|
|
node.attributes.update(attrs)
|
|
|
|
node += contnode
|
|
|
|
return node, contnode
|
|
|
|
|
|
|
|
def reference_check(*args, **kwds):
|
|
|
|
node, contnode = fake_node(*args, **kwds)
|
|
|
|
return missing_reference(app, app.env, node, contnode)
|
|
|
|
|
|
|
|
# check resolution when a target is found
|
|
|
|
rn = reference_check('py', 'func', 'module1.func', 'foo')
|
2009-09-09 14:56:53 -05:00
|
|
|
assert isinstance(rn, nodes.reference)
|
2015-02-22 22:20:35 -06:00
|
|
|
assert rn['refuri'] == 'https://docs.python.org/sub/foo.html#module1.func'
|
2009-09-09 14:56:53 -05:00
|
|
|
assert rn['reftitle'] == '(in foo v2.0)'
|
2010-08-05 04:58:43 -05:00
|
|
|
assert rn[0].astext() == 'foo'
|
2009-09-09 14:56:53 -05:00
|
|
|
|
|
|
|
# create unresolvable nodes and check None return value
|
2010-08-05 04:58:43 -05:00
|
|
|
assert reference_check('py', 'foo', 'module1.func', 'foo') is None
|
|
|
|
assert reference_check('py', 'func', 'foo', 'foo') is None
|
|
|
|
assert reference_check('py', 'func', 'foo', 'foo') is None
|
2010-07-27 06:20:58 -05:00
|
|
|
|
|
|
|
# check handling of prefixes
|
|
|
|
|
|
|
|
# prefix given, target found: prefix is stripped
|
2010-08-05 04:58:43 -05:00
|
|
|
rn = reference_check('py', 'mod', 'py3k:module2', 'py3k:module2')
|
|
|
|
assert rn[0].astext() == 'module2'
|
|
|
|
|
|
|
|
# prefix given, but not in title: nothing stripped
|
|
|
|
rn = reference_check('py', 'mod', 'py3k:module2', 'module2')
|
2010-07-27 06:20:58 -05:00
|
|
|
assert rn[0].astext() == 'module2'
|
|
|
|
|
2010-08-05 04:58:43 -05:00
|
|
|
# prefix given, but explicit: nothing stripped
|
|
|
|
rn = reference_check('py', 'mod', 'py3k:module2', 'py3k:module2',
|
|
|
|
refexplicit=True)
|
|
|
|
assert rn[0].astext() == 'py3k:module2'
|
|
|
|
|
2010-07-27 06:20:58 -05:00
|
|
|
# prefix given, target not found and nonexplicit title: prefix is stripped
|
2010-08-05 04:58:43 -05:00
|
|
|
node, contnode = fake_node('py', 'mod', 'py3k:unknown', 'py3k:unknown',
|
|
|
|
refexplicit=False)
|
|
|
|
rn = missing_reference(app, app.env, node, contnode)
|
2010-07-27 06:20:58 -05:00
|
|
|
assert rn is None
|
|
|
|
assert contnode[0].astext() == 'unknown'
|
|
|
|
|
|
|
|
# prefix given, target not found and explicit title: nothing is changed
|
2010-08-05 04:58:43 -05:00
|
|
|
node, contnode = fake_node('py', 'mod', 'py3k:unknown', 'py3k:unknown',
|
|
|
|
refexplicit=True)
|
|
|
|
rn = missing_reference(app, app.env, node, contnode)
|
2010-07-27 06:20:58 -05:00
|
|
|
assert rn is None
|
|
|
|
assert contnode[0].astext() == 'py3k:unknown'
|
2011-09-03 14:41:26 -05:00
|
|
|
|
|
|
|
|
2014-01-10 15:10:29 -06:00
|
|
|
@with_app()
|
2011-09-03 14:41:26 -05:00
|
|
|
@with_tempdir
|
2014-09-21 10:17:02 -05:00
|
|
|
def test_load_mappings_warnings(tempdir, app, status, warning):
|
2011-09-03 14:41:26 -05:00
|
|
|
"""
|
|
|
|
load_mappings issues a warning if new-style mapping
|
2016-05-22 22:14:56 -05:00
|
|
|
identifiers are not string
|
2011-09-03 14:41:26 -05:00
|
|
|
"""
|
|
|
|
inv_file = tempdir / 'inventory'
|
2014-04-28 05:58:26 -05:00
|
|
|
inv_file.write_bytes(inventory_v2)
|
2011-09-03 14:41:26 -05:00
|
|
|
app.config.intersphinx_mapping = {
|
2015-02-22 22:20:35 -06:00
|
|
|
'https://docs.python.org/': inv_file,
|
|
|
|
'py3k': ('https://docs.python.org/py3k/', inv_file),
|
2011-09-03 14:41:26 -05:00
|
|
|
'repoze.workflow': ('http://docs.repoze.org/workflow/', inv_file),
|
2011-10-09 16:09:57 -05:00
|
|
|
'django-taggit': ('http://django-taggit.readthedocs.org/en/latest/',
|
2016-05-22 22:11:10 -05:00
|
|
|
inv_file),
|
|
|
|
12345: ('http://www.sphinx-doc.org/en/stable/', inv_file),
|
2011-09-03 14:41:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
app.config.intersphinx_cache_limit = 0
|
|
|
|
# load the inventory and check if it's done correctly
|
|
|
|
load_mappings(app)
|
2016-05-22 22:14:56 -05:00
|
|
|
assert warning.getvalue().count('\n') == 1
|
2015-10-15 16:37:55 -05:00
|
|
|
|
|
|
|
|
|
|
|
class TestStripBasicAuth(unittest.TestCase):
|
|
|
|
"""Tests for sphinx.ext.intersphinx._strip_basic_auth()"""
|
|
|
|
def test_auth_stripped(self):
|
|
|
|
"""basic auth creds stripped from URL containing creds"""
|
|
|
|
url = 'https://user:12345@domain.com/project/objects.inv'
|
|
|
|
expected = 'https://domain.com/project/objects.inv'
|
|
|
|
actual_url, actual_username, actual_password = _strip_basic_auth(url)
|
|
|
|
self.assertEqual(expected, actual_url)
|
|
|
|
self.assertEqual('user', actual_username)
|
|
|
|
self.assertEqual('12345', actual_password)
|
|
|
|
|
|
|
|
def test_no_auth(self):
|
|
|
|
"""url unchanged if param doesn't contain basic auth creds"""
|
|
|
|
url = 'https://domain.com/project/objects.inv'
|
|
|
|
expected = 'https://domain.com/project/objects.inv'
|
|
|
|
actual_url, actual_username, actual_password = _strip_basic_auth(url)
|
|
|
|
self.assertEqual(expected, actual_url)
|
|
|
|
self.assertEqual(None, actual_username)
|
|
|
|
self.assertEqual(None, actual_password)
|
|
|
|
|
2016-04-24 21:08:03 -05:00
|
|
|
def test_having_port(self):
|
|
|
|
"""basic auth creds correctly stripped from URL containing creds even if URL
|
|
|
|
contains port"""
|
|
|
|
url = 'https://user:12345@domain.com:8080/project/objects.inv'
|
|
|
|
expected = 'https://domain.com:8080/project/objects.inv'
|
|
|
|
actual_url, actual_username, actual_password = _strip_basic_auth(url)
|
|
|
|
self.assertEqual(expected, actual_url)
|
|
|
|
self.assertEqual('user', actual_username)
|
|
|
|
self.assertEqual('12345', actual_password)
|
|
|
|
|
2015-10-15 16:37:55 -05:00
|
|
|
|
2015-10-22 01:09:16 -05:00
|
|
|
def test_getsafeurl_authed():
|
|
|
|
"""_get_safe_url() with a url with basic auth"""
|
|
|
|
url = 'https://user:12345@domain.com/project/objects.inv'
|
|
|
|
expected = 'https://user@domain.com/project/objects.inv'
|
|
|
|
actual = _get_safe_url(url)
|
|
|
|
assert expected == actual
|
|
|
|
|
|
|
|
|
2016-08-17 11:25:29 -05:00
|
|
|
def test_getsafeurl_authed_having_port():
|
|
|
|
"""_get_safe_url() with a url with basic auth having port"""
|
|
|
|
url = 'https://user:12345@domain.com:8080/project/objects.inv'
|
|
|
|
expected = 'https://user@domain.com:8080/project/objects.inv'
|
|
|
|
actual = _get_safe_url(url)
|
|
|
|
assert expected == actual
|
|
|
|
|
|
|
|
|
2015-10-22 01:09:16 -05:00
|
|
|
def test_getsafeurl_unauthed():
|
|
|
|
"""_get_safe_url() with a url without basic auth"""
|
|
|
|
url = 'https://domain.com/project/objects.inv'
|
|
|
|
expected = 'https://domain.com/project/objects.inv'
|
|
|
|
actual = _get_safe_url(url)
|
|
|
|
assert expected == actual
|