urlresolvers.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576
  1. """
  2. This module converts requested URLs to callback view functions.
  3. RegexURLResolver is the main class here. Its resolve() method takes a URL (as
  4. a string) and returns a tuple in this format:
  5. (view_function, function_args, function_kwargs)
  6. """
  7. from __future__ import unicode_literals
  8. from importlib import import_module
  9. import re
  10. from threading import local
  11. from django.http import Http404
  12. from django.core.exceptions import ImproperlyConfigured, ViewDoesNotExist
  13. from django.utils.datastructures import MultiValueDict
  14. from django.utils.encoding import force_str, force_text, iri_to_uri
  15. from django.utils.functional import memoize, lazy
  16. from django.utils.http import urlquote
  17. from django.utils.module_loading import module_has_submodule
  18. from django.utils.regex_helper import normalize
  19. from django.utils import six
  20. from django.utils.translation import get_language
  21. _resolver_cache = {} # Maps URLconf modules to RegexURLResolver instances.
  22. _ns_resolver_cache = {} # Maps namespaces to RegexURLResolver instances.
  23. _callable_cache = {} # Maps view and url pattern names to their view functions.
  24. # SCRIPT_NAME prefixes for each thread are stored here. If there's no entry for
  25. # the current thread (which is the only one we ever access), it is assumed to
  26. # be empty.
  27. _prefixes = local()
  28. # Overridden URLconfs for each thread are stored here.
  29. _urlconfs = local()
  30. class ResolverMatch(object):
  31. def __init__(self, func, args, kwargs, url_name=None, app_name=None, namespaces=None):
  32. self.func = func
  33. self.args = args
  34. self.kwargs = kwargs
  35. self.app_name = app_name
  36. if namespaces:
  37. self.namespaces = [x for x in namespaces if x]
  38. else:
  39. self.namespaces = []
  40. if not url_name:
  41. if not hasattr(func, '__name__'):
  42. # An instance of a callable class
  43. url_name = '.'.join([func.__class__.__module__, func.__class__.__name__])
  44. else:
  45. # A function
  46. url_name = '.'.join([func.__module__, func.__name__])
  47. self.url_name = url_name
  48. @property
  49. def namespace(self):
  50. return ':'.join(self.namespaces)
  51. @property
  52. def view_name(self):
  53. return ':'.join(filter(bool, (self.namespace, self.url_name)))
  54. def __getitem__(self, index):
  55. return (self.func, self.args, self.kwargs)[index]
  56. def __repr__(self):
  57. return "ResolverMatch(func=%s, args=%s, kwargs=%s, url_name='%s', app_name='%s', namespace='%s')" % (
  58. self.func, self.args, self.kwargs, self.url_name, self.app_name, self.namespace)
  59. class Resolver404(Http404):
  60. pass
  61. class NoReverseMatch(Exception):
  62. pass
  63. def get_callable(lookup_view, can_fail=False):
  64. """
  65. Convert a string version of a function name to the callable object.
  66. If the lookup_view is not an import path, it is assumed to be a URL pattern
  67. label and the original string is returned.
  68. If can_fail is True, lookup_view might be a URL pattern label, so errors
  69. during the import fail and the string is returned.
  70. """
  71. if not callable(lookup_view):
  72. mod_name, func_name = get_mod_func(lookup_view)
  73. if func_name == '':
  74. return lookup_view
  75. try:
  76. mod = import_module(mod_name)
  77. except ImportError:
  78. parentmod, submod = get_mod_func(mod_name)
  79. if (not can_fail and submod != '' and
  80. not module_has_submodule(import_module(parentmod), submod)):
  81. raise ViewDoesNotExist(
  82. "Could not import %s. Parent module %s does not exist." %
  83. (lookup_view, mod_name))
  84. if not can_fail:
  85. raise
  86. else:
  87. try:
  88. lookup_view = getattr(mod, func_name)
  89. if not callable(lookup_view):
  90. raise ViewDoesNotExist(
  91. "Could not import %s.%s. View is not callable." %
  92. (mod_name, func_name))
  93. except AttributeError:
  94. if not can_fail:
  95. raise ViewDoesNotExist(
  96. "Could not import %s. View does not exist in module %s." %
  97. (lookup_view, mod_name))
  98. return lookup_view
  99. get_callable = memoize(get_callable, _callable_cache, 1)
  100. def get_resolver(urlconf):
  101. if urlconf is None:
  102. from django.conf import settings
  103. urlconf = settings.ROOT_URLCONF
  104. return RegexURLResolver(r'^/', urlconf)
  105. get_resolver = memoize(get_resolver, _resolver_cache, 1)
  106. def get_ns_resolver(ns_pattern, resolver):
  107. # Build a namespaced resolver for the given parent urlconf pattern.
  108. # This makes it possible to have captured parameters in the parent
  109. # urlconf pattern.
  110. ns_resolver = RegexURLResolver(ns_pattern,
  111. resolver.url_patterns)
  112. return RegexURLResolver(r'^/', [ns_resolver])
  113. get_ns_resolver = memoize(get_ns_resolver, _ns_resolver_cache, 2)
  114. def get_mod_func(callback):
  115. # Converts 'django.views.news.stories.story_detail' to
  116. # ['django.views.news.stories', 'story_detail']
  117. try:
  118. dot = callback.rindex('.')
  119. except ValueError:
  120. return callback, ''
  121. return callback[:dot], callback[dot+1:]
  122. class LocaleRegexProvider(object):
  123. """
  124. A mixin to provide a default regex property which can vary by active
  125. language.
  126. """
  127. def __init__(self, regex):
  128. # regex is either a string representing a regular expression, or a
  129. # translatable string (using ugettext_lazy) representing a regular
  130. # expression.
  131. self._regex = regex
  132. self._regex_dict = {}
  133. @property
  134. def regex(self):
  135. """
  136. Returns a compiled regular expression, depending upon the activated
  137. language-code.
  138. """
  139. language_code = get_language()
  140. if language_code not in self._regex_dict:
  141. if isinstance(self._regex, six.string_types):
  142. regex = self._regex
  143. else:
  144. regex = force_text(self._regex)
  145. try:
  146. compiled_regex = re.compile(regex, re.UNICODE)
  147. except re.error as e:
  148. raise ImproperlyConfigured(
  149. '"%s" is not a valid regular expression: %s' %
  150. (regex, six.text_type(e)))
  151. self._regex_dict[language_code] = compiled_regex
  152. return self._regex_dict[language_code]
  153. class RegexURLPattern(LocaleRegexProvider):
  154. def __init__(self, regex, callback, default_args=None, name=None):
  155. LocaleRegexProvider.__init__(self, regex)
  156. # callback is either a string like 'foo.views.news.stories.story_detail'
  157. # which represents the path to a module and a view function name, or a
  158. # callable object (view).
  159. if callable(callback):
  160. self._callback = callback
  161. else:
  162. self._callback = None
  163. self._callback_str = callback
  164. self.default_args = default_args or {}
  165. self.name = name
  166. def __repr__(self):
  167. return force_str('<%s %s %s>' % (self.__class__.__name__, self.name, self.regex.pattern))
  168. def add_prefix(self, prefix):
  169. """
  170. Adds the prefix string to a string-based callback.
  171. """
  172. if not prefix or not hasattr(self, '_callback_str'):
  173. return
  174. self._callback_str = prefix + '.' + self._callback_str
  175. def resolve(self, path):
  176. match = self.regex.search(path)
  177. if match:
  178. # If there are any named groups, use those as kwargs, ignoring
  179. # non-named groups. Otherwise, pass all non-named arguments as
  180. # positional arguments.
  181. kwargs = match.groupdict()
  182. if kwargs:
  183. args = ()
  184. else:
  185. args = match.groups()
  186. # In both cases, pass any extra_kwargs as **kwargs.
  187. kwargs.update(self.default_args)
  188. return ResolverMatch(self.callback, args, kwargs, self.name)
  189. @property
  190. def callback(self):
  191. if self._callback is not None:
  192. return self._callback
  193. self._callback = get_callable(self._callback_str)
  194. return self._callback
  195. class RegexURLResolver(LocaleRegexProvider):
  196. def __init__(self, regex, urlconf_name, default_kwargs=None, app_name=None, namespace=None):
  197. LocaleRegexProvider.__init__(self, regex)
  198. # urlconf_name is a string representing the module containing URLconfs.
  199. self.urlconf_name = urlconf_name
  200. if not isinstance(urlconf_name, six.string_types):
  201. self._urlconf_module = self.urlconf_name
  202. self.callback = None
  203. self.default_kwargs = default_kwargs or {}
  204. self.namespace = namespace
  205. self.app_name = app_name
  206. self._reverse_dict = {}
  207. self._namespace_dict = {}
  208. self._app_dict = {}
  209. def __repr__(self):
  210. if isinstance(self.urlconf_name, list) and len(self.urlconf_name):
  211. # Don't bother to output the whole list, it can be huge
  212. urlconf_repr = '<%s list>' % self.urlconf_name[0].__class__.__name__
  213. else:
  214. urlconf_repr = repr(self.urlconf_name)
  215. return str('<%s %s (%s:%s) %s>') % (
  216. self.__class__.__name__, urlconf_repr, self.app_name,
  217. self.namespace, self.regex.pattern)
  218. def _populate(self):
  219. lookups = MultiValueDict()
  220. namespaces = {}
  221. apps = {}
  222. language_code = get_language()
  223. for pattern in reversed(self.url_patterns):
  224. p_pattern = pattern.regex.pattern
  225. if p_pattern.startswith('^'):
  226. p_pattern = p_pattern[1:]
  227. if isinstance(pattern, RegexURLResolver):
  228. if pattern.namespace:
  229. namespaces[pattern.namespace] = (p_pattern, pattern)
  230. if pattern.app_name:
  231. apps.setdefault(pattern.app_name, []).append(pattern.namespace)
  232. else:
  233. parent = normalize(pattern.regex.pattern)
  234. for name in pattern.reverse_dict:
  235. for matches, pat, defaults in pattern.reverse_dict.getlist(name):
  236. new_matches = []
  237. for piece, p_args in parent:
  238. new_matches.extend((piece + suffix, p_args + args) for (suffix, args) in matches)
  239. lookups.appendlist(name, (new_matches, p_pattern + pat, dict(defaults, **pattern.default_kwargs)))
  240. for namespace, (prefix, sub_pattern) in pattern.namespace_dict.items():
  241. namespaces[namespace] = (p_pattern + prefix, sub_pattern)
  242. for app_name, namespace_list in pattern.app_dict.items():
  243. apps.setdefault(app_name, []).extend(namespace_list)
  244. else:
  245. bits = normalize(p_pattern)
  246. lookups.appendlist(pattern.callback, (bits, p_pattern, pattern.default_args))
  247. if pattern.name is not None:
  248. lookups.appendlist(pattern.name, (bits, p_pattern, pattern.default_args))
  249. self._reverse_dict[language_code] = lookups
  250. self._namespace_dict[language_code] = namespaces
  251. self._app_dict[language_code] = apps
  252. @property
  253. def reverse_dict(self):
  254. language_code = get_language()
  255. if language_code not in self._reverse_dict:
  256. self._populate()
  257. return self._reverse_dict[language_code]
  258. @property
  259. def namespace_dict(self):
  260. language_code = get_language()
  261. if language_code not in self._namespace_dict:
  262. self._populate()
  263. return self._namespace_dict[language_code]
  264. @property
  265. def app_dict(self):
  266. language_code = get_language()
  267. if language_code not in self._app_dict:
  268. self._populate()
  269. return self._app_dict[language_code]
  270. def resolve(self, path):
  271. path = force_text(path) # path may be a reverse_lazy object
  272. tried = []
  273. match = self.regex.search(path)
  274. if match:
  275. new_path = path[match.end():]
  276. for pattern in self.url_patterns:
  277. try:
  278. sub_match = pattern.resolve(new_path)
  279. except Resolver404 as e:
  280. sub_tried = e.args[0].get('tried')
  281. if sub_tried is not None:
  282. tried.extend([pattern] + t for t in sub_tried)
  283. else:
  284. tried.append([pattern])
  285. else:
  286. if sub_match:
  287. sub_match_dict = dict(match.groupdict(), **self.default_kwargs)
  288. sub_match_dict.update(sub_match.kwargs)
  289. return ResolverMatch(sub_match.func, sub_match.args, sub_match_dict, sub_match.url_name, self.app_name or sub_match.app_name, [self.namespace] + sub_match.namespaces)
  290. tried.append([pattern])
  291. raise Resolver404({'tried': tried, 'path': new_path})
  292. raise Resolver404({'path' : path})
  293. @property
  294. def urlconf_module(self):
  295. try:
  296. return self._urlconf_module
  297. except AttributeError:
  298. self._urlconf_module = import_module(self.urlconf_name)
  299. return self._urlconf_module
  300. @property
  301. def url_patterns(self):
  302. patterns = getattr(self.urlconf_module, "urlpatterns", self.urlconf_module)
  303. try:
  304. iter(patterns)
  305. except TypeError:
  306. raise ImproperlyConfigured("The included urlconf %s doesn't have any patterns in it" % self.urlconf_name)
  307. return patterns
  308. def _resolve_special(self, view_type):
  309. callback = getattr(self.urlconf_module, 'handler%s' % view_type, None)
  310. if not callback:
  311. # No handler specified in file; use default
  312. # Lazy import, since django.urls imports this file
  313. from django.conf import urls
  314. callback = getattr(urls, 'handler%s' % view_type)
  315. return get_callable(callback), {}
  316. def resolve400(self):
  317. return self._resolve_special('400')
  318. def resolve403(self):
  319. return self._resolve_special('403')
  320. def resolve404(self):
  321. return self._resolve_special('404')
  322. def resolve500(self):
  323. return self._resolve_special('500')
  324. def reverse(self, lookup_view, *args, **kwargs):
  325. return self._reverse_with_prefix(lookup_view, '', *args, **kwargs)
  326. def _reverse_with_prefix(self, lookup_view, _prefix, *args, **kwargs):
  327. if args and kwargs:
  328. raise ValueError("Don't mix *args and **kwargs in call to reverse()!")
  329. text_args = [force_text(v) for v in args]
  330. text_kwargs = dict((k, force_text(v)) for (k, v) in kwargs.items())
  331. try:
  332. lookup_view = get_callable(lookup_view, True)
  333. except (ImportError, AttributeError) as e:
  334. raise NoReverseMatch("Error importing '%s': %s." % (lookup_view, e))
  335. possibilities = self.reverse_dict.getlist(lookup_view)
  336. prefix_norm, prefix_args = normalize(urlquote(_prefix))[0]
  337. for possibility, pattern, defaults in possibilities:
  338. for result, params in possibility:
  339. if args:
  340. if len(args) != len(params) + len(prefix_args):
  341. continue
  342. candidate_subs = dict(zip(prefix_args + params, text_args))
  343. else:
  344. if set(kwargs.keys()) | set(defaults.keys()) != set(params) | set(defaults.keys()) | set(prefix_args):
  345. continue
  346. matches = True
  347. for k, v in defaults.items():
  348. if kwargs.get(k, v) != v:
  349. matches = False
  350. break
  351. if not matches:
  352. continue
  353. candidate_subs = text_kwargs
  354. # WSGI provides decoded URLs, without %xx escapes, and the URL
  355. # resolver operates on such URLs. First substitute arguments
  356. # without quoting to build a decoded URL and look for a match.
  357. # Then, if we have a match, redo the substitution with quoted
  358. # arguments in order to return a properly encoded URL.
  359. candidate_pat = prefix_norm.replace('%', '%%') + result
  360. if re.search('^%s%s' % (prefix_norm, pattern), candidate_pat % candidate_subs, re.UNICODE):
  361. candidate_subs = dict((k, urlquote(v)) for (k, v) in candidate_subs.items())
  362. return candidate_pat % candidate_subs
  363. # lookup_view can be URL label, or dotted path, or callable, Any of
  364. # these can be passed in at the top, but callables are not friendly in
  365. # error messages.
  366. m = getattr(lookup_view, '__module__', None)
  367. n = getattr(lookup_view, '__name__', None)
  368. if m is not None and n is not None:
  369. lookup_view_s = "%s.%s" % (m, n)
  370. else:
  371. lookup_view_s = lookup_view
  372. patterns = [pattern for (possibility, pattern, defaults) in possibilities]
  373. raise NoReverseMatch("Reverse for '%s' with arguments '%s' and keyword "
  374. "arguments '%s' not found. %d pattern(s) tried: %s" %
  375. (lookup_view_s, args, kwargs, len(patterns), patterns))
  376. class LocaleRegexURLResolver(RegexURLResolver):
  377. """
  378. A URL resolver that always matches the active language code as URL prefix.
  379. Rather than taking a regex argument, we just override the ``regex``
  380. function to always return the active language-code as regex.
  381. """
  382. def __init__(self, urlconf_name, default_kwargs=None, app_name=None, namespace=None):
  383. super(LocaleRegexURLResolver, self).__init__(
  384. None, urlconf_name, default_kwargs, app_name, namespace)
  385. @property
  386. def regex(self):
  387. language_code = get_language()
  388. if language_code not in self._regex_dict:
  389. regex_compiled = re.compile('^%s/' % language_code, re.UNICODE)
  390. self._regex_dict[language_code] = regex_compiled
  391. return self._regex_dict[language_code]
  392. def resolve(path, urlconf=None):
  393. if urlconf is None:
  394. urlconf = get_urlconf()
  395. return get_resolver(urlconf).resolve(path)
  396. def reverse(viewname, urlconf=None, args=None, kwargs=None, prefix=None, current_app=None):
  397. if urlconf is None:
  398. urlconf = get_urlconf()
  399. resolver = get_resolver(urlconf)
  400. args = args or []
  401. kwargs = kwargs or {}
  402. if prefix is None:
  403. prefix = get_script_prefix()
  404. if not isinstance(viewname, six.string_types):
  405. view = viewname
  406. else:
  407. parts = viewname.split(':')
  408. parts.reverse()
  409. view = parts[0]
  410. path = parts[1:]
  411. resolved_path = []
  412. ns_pattern = ''
  413. while path:
  414. ns = path.pop()
  415. # Lookup the name to see if it could be an app identifier
  416. try:
  417. app_list = resolver.app_dict[ns]
  418. # Yes! Path part matches an app in the current Resolver
  419. if current_app and current_app in app_list:
  420. # If we are reversing for a particular app,
  421. # use that namespace
  422. ns = current_app
  423. elif ns not in app_list:
  424. # The name isn't shared by one of the instances
  425. # (i.e., the default) so just pick the first instance
  426. # as the default.
  427. ns = app_list[0]
  428. except KeyError:
  429. pass
  430. try:
  431. extra, resolver = resolver.namespace_dict[ns]
  432. resolved_path.append(ns)
  433. ns_pattern = ns_pattern + extra
  434. except KeyError as key:
  435. if resolved_path:
  436. raise NoReverseMatch(
  437. "%s is not a registered namespace inside '%s'" %
  438. (key, ':'.join(resolved_path)))
  439. else:
  440. raise NoReverseMatch("%s is not a registered namespace" %
  441. key)
  442. if ns_pattern:
  443. resolver = get_ns_resolver(ns_pattern, resolver)
  444. return iri_to_uri(resolver._reverse_with_prefix(view, prefix, *args, **kwargs))
  445. reverse_lazy = lazy(reverse, str)
  446. def clear_url_caches():
  447. global _resolver_cache
  448. global _ns_resolver_cache
  449. global _callable_cache
  450. _resolver_cache.clear()
  451. _ns_resolver_cache.clear()
  452. _callable_cache.clear()
  453. def set_script_prefix(prefix):
  454. """
  455. Sets the script prefix for the current thread.
  456. """
  457. if not prefix.endswith('/'):
  458. prefix += '/'
  459. _prefixes.value = prefix
  460. def get_script_prefix():
  461. """
  462. Returns the currently active script prefix. Useful for client code that
  463. wishes to construct their own URLs manually (although accessing the request
  464. instance is normally going to be a lot cleaner).
  465. """
  466. return getattr(_prefixes, "value", '/')
  467. def clear_script_prefix():
  468. """
  469. Unsets the script prefix for the current thread.
  470. """
  471. try:
  472. del _prefixes.value
  473. except AttributeError:
  474. pass
  475. def set_urlconf(urlconf_name):
  476. """
  477. Sets the URLconf for the current thread (overriding the default one in
  478. settings). Set to None to revert back to the default.
  479. """
  480. if urlconf_name:
  481. _urlconfs.value = urlconf_name
  482. else:
  483. if hasattr(_urlconfs, "value"):
  484. del _urlconfs.value
  485. def get_urlconf(default=None):
  486. """
  487. Returns the root URLconf to use for the current thread if it has been
  488. changed from the default one.
  489. """
  490. return getattr(_urlconfs, "value", default)
  491. def is_valid_path(path, urlconf=None):
  492. """
  493. Returns True if the given path resolves against the default URL resolver,
  494. False otherwise.
  495. This is a convenience method to make working with "is this a match?" cases
  496. easier, avoiding unnecessarily indented try...except blocks.
  497. """
  498. try:
  499. resolve(path, urlconf)
  500. return True
  501. except Resolver404:
  502. return False