2013-04-05 00:42:26 +04:00
|
|
|
"""
|
|
|
|
Utility functions to return a formatted name and description for a given view.
|
|
|
|
"""
|
2015-06-18 16:38:29 +03:00
|
|
|
import re
|
|
|
|
|
2015-08-07 00:51:35 +03:00
|
|
|
from django.utils.encoding import force_text
|
2013-04-05 00:42:26 +04:00
|
|
|
from django.utils.html import escape
|
|
|
|
from django.utils.safestring import mark_safe
|
2015-06-18 16:38:29 +03:00
|
|
|
|
2015-08-07 00:51:35 +03:00
|
|
|
from rest_framework.compat import apply_markdown
|
2013-04-05 00:42:26 +04:00
|
|
|
|
|
|
|
|
2013-08-19 11:24:27 +04:00
|
|
|
def remove_trailing_string(content, trailing):
|
2013-04-05 00:42:26 +04:00
|
|
|
"""
|
|
|
|
Strip trailing component `trailing` from `content` if it exists.
|
|
|
|
Used when generating names from view classes.
|
|
|
|
"""
|
|
|
|
if content.endswith(trailing) and content != trailing:
|
|
|
|
return content[:-len(trailing)]
|
|
|
|
return content
|
|
|
|
|
|
|
|
|
2013-08-19 11:24:27 +04:00
|
|
|
def dedent(content):
|
2013-04-05 00:42:26 +04:00
|
|
|
"""
|
|
|
|
Remove leading indent from a block of text.
|
|
|
|
Used when generating descriptions from docstrings.
|
2013-08-19 11:24:27 +04:00
|
|
|
|
|
|
|
Note that python's `textwrap.dedent` doesn't quite cut it,
|
|
|
|
as it fails to dedent multiline docstrings that include
|
|
|
|
unindented text on the initial line.
|
2013-04-05 00:42:26 +04:00
|
|
|
"""
|
2014-08-19 05:09:48 +04:00
|
|
|
content = force_text(content)
|
2017-01-31 22:57:52 +03:00
|
|
|
lines = [line for line in content.splitlines()[1:] if line.lstrip()]
|
2013-04-05 00:42:26 +04:00
|
|
|
|
|
|
|
# unindent the content if needed
|
2017-01-31 22:57:52 +03:00
|
|
|
if lines:
|
|
|
|
whitespace_counts = min([len(line) - len(line.lstrip(' ')) for line in lines])
|
|
|
|
tab_counts = min([len(line) - len(line.lstrip('\t')) for line in lines])
|
|
|
|
if whitespace_counts:
|
|
|
|
whitespace_pattern = '^' + (' ' * whitespace_counts)
|
|
|
|
content = re.sub(re.compile(whitespace_pattern, re.MULTILINE), '', content)
|
|
|
|
elif tab_counts:
|
|
|
|
whitespace_pattern = '^' + ('\t' * tab_counts)
|
|
|
|
content = re.sub(re.compile(whitespace_pattern, re.MULTILINE), '', content)
|
2013-08-19 11:24:27 +04:00
|
|
|
return content.strip()
|
2013-04-05 00:42:26 +04:00
|
|
|
|
2014-08-19 16:28:07 +04:00
|
|
|
|
2013-08-19 11:24:27 +04:00
|
|
|
def camelcase_to_spaces(content):
|
2013-04-05 00:42:26 +04:00
|
|
|
"""
|
|
|
|
Translate 'CamelCaseNames' to 'Camel Case Names'.
|
|
|
|
Used when generating names from view classes.
|
|
|
|
"""
|
2017-07-16 19:12:29 +03:00
|
|
|
camelcase_boundary = '(((?<=[a-z])[A-Z])|([A-Z](?![A-Z]|$)))'
|
|
|
|
content = re.sub(camelcase_boundary, ' \\1', content).strip()
|
2013-04-05 00:42:26 +04:00
|
|
|
return ' '.join(content.split('_')).title()
|
|
|
|
|
2014-08-19 16:28:07 +04:00
|
|
|
|
2013-04-05 00:42:26 +04:00
|
|
|
def markup_description(description):
|
|
|
|
"""
|
|
|
|
Apply HTML markup to the given description.
|
|
|
|
"""
|
|
|
|
if apply_markdown:
|
|
|
|
description = apply_markdown(description)
|
|
|
|
else:
|
|
|
|
description = escape(description).replace('\n', '<br />')
|
2015-01-14 16:19:56 +03:00
|
|
|
description = '<p>' + description + '</p>'
|
2013-04-05 00:42:26 +04:00
|
|
|
return mark_safe(description)
|
2019-05-29 21:32:03 +03:00
|
|
|
|
|
|
|
|
|
|
|
class lazy_format:
|
|
|
|
"""
|
|
|
|
Delay formatting until it's actually needed.
|
|
|
|
|
|
|
|
Useful when the format string or one of the arguments is lazy.
|
|
|
|
|
|
|
|
Not using Django's lazy because it is too slow.
|
|
|
|
"""
|
|
|
|
__slots__ = ('format_string', 'args', 'kwargs', 'result')
|
|
|
|
|
|
|
|
def __init__(self, format_string, *args, **kwargs):
|
|
|
|
self.result = None
|
|
|
|
self.format_string = format_string
|
|
|
|
self.args = args
|
|
|
|
self.kwargs = kwargs
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
if self.result is None:
|
|
|
|
self.result = self.format_string.format(*self.args, **self.kwargs)
|
|
|
|
self.format_string, self.args, self.kwargs = None, None, None
|
|
|
|
return self.result
|
|
|
|
|
|
|
|
def __mod__(self, value):
|
|
|
|
return str(self) % value
|