Compare commits

...

13 Commits

Author SHA1 Message Date
AlexVonB
d375116807 Merge branch 'develop' 2022-04-13 20:47:52 +02:00
AlexVonB
87b9f6c88e bump to v0.11.0 2022-04-13 20:47:30 +02:00
AlexVonB
bda367dad9 Merge branch 'tdgroot-code_language_callback' into develop
closes #64
2022-04-13 20:44:18 +02:00
AlexVonB
61e8940486 added readme for callback 2022-04-13 20:42:38 +02:00
AlexVonB
35479d2d3b Merge branch 'code_language_callback' of https://github.com/tdgroot/python-markdownify into tdgroot-code_language_callback 2022-04-13 20:25:37 +02:00
AlexVonB
b589863715 add escaping of asterisks and option to disable it
closes #62
2022-04-13 20:04:12 +02:00
AlexVonB
423b7e948c add option to allow inline images in selected tags
fixes #61
2022-04-13 19:55:34 +02:00
Timon de Groot
0ea95de4d0 Add code language callback 2022-04-09 13:22:28 +02:00
AlexVonB
ed3eee78d2 fixed readme 2022-01-24 18:18:19 +01:00
AlexVonB
eb0330bfc6 Merge branch 'develop' 2022-01-23 11:01:45 +01:00
AlexVonB
ddda696396 bump to v0.10.3 2022-01-23 11:01:26 +01:00
AlexVonB
0a1343a538 allow BeautifulSoup objects to be converted 2022-01-23 11:00:19 +01:00
AlexVonB
9d0b839b73 wording 2022-01-23 10:59:24 +01:00
6 changed files with 92 additions and 21 deletions

View File

@@ -32,14 +32,14 @@ Convert some HTML to Markdown:
from markdownify import markdownify as md from markdownify import markdownify as md
md('<b>Yay</b> <a href="http://github.com">GitHub</a>') # > '**Yay** [GitHub](http://github.com)' md('<b>Yay</b> <a href="http://github.com">GitHub</a>') # > '**Yay** [GitHub](http://github.com)'
Specify tags to exclude (blacklist): Specify tags to exclude:
.. code:: python .. code:: python
from markdownify import markdownify as md from markdownify import markdownify as md
md('<b>Yay</b> <a href="http://github.com">GitHub</a>', strip=['a']) # > '**Yay** GitHub' md('<b>Yay</b> <a href="http://github.com">GitHub</a>', strip=['a']) # > '**Yay** GitHub'
\...or specify the tags you want to include (whitelist): \...or specify the tags you want to include:
.. code:: python .. code:: python
@@ -53,11 +53,11 @@ Options
Markdownify supports the following options: Markdownify supports the following options:
strip strip
A list of tags to strip (blacklist). This option can't be used with the A list of tags to strip. This option can't be used with the
``convert`` option. ``convert`` option.
convert convert
A list of tags to convert (whitelist). This option can't be used with the A list of tags to convert. This option can't be used with the
``strip`` option. ``strip`` option.
autolinks autolinks
@@ -102,14 +102,50 @@ code_language
should be annotated with `````python`` or similar. should be annotated with `````python`` or similar.
Defaults to ``''`` (empty string) and can be any string. Defaults to ``''`` (empty string) and can be any string.
code_language_callback
When the HTML code contains ``pre`` tags that in some way provide the code
language, for example as class, this callback can be used to extract the
language from the tag and prefix it to the converted ``pre`` tag.
The callback gets one single argument, an BeautifylSoup object, and returns
a string containing the code language, or ``None``.
An example to use the class name as code language could be::
def callback(el):
return el['class'][0] if el.has_attr('class') else None
Defaults to ``None``.
escape_asterisks
If set to ``False``, do not escape ``*`` to ``\*`` in text.
Defaults to ``True``.
escape_underscores escape_underscores
If set to ``False``, do not escape ``_`` to ``\_`` in text. If set to ``False``, do not escape ``_`` to ``\_`` in text.
Defaults to ``True``. Defaults to ``True``.
keep_inline_images_in
Images are converted to their alt-text when the images are located inside
headlines or table cells. If some inline images should be converted to
markdown images instead, this option can be set to a list of parent tags
that should be allowed to contain inline images, for example ``['td']``.
Defaults to an empty list.
Options may be specified as kwargs to the ``markdownify`` function, or as a Options may be specified as kwargs to the ``markdownify`` function, or as a
nested ``Options`` class in ``MarkdownConverter`` subclasses. nested ``Options`` class in ``MarkdownConverter`` subclasses.
Converting BeautifulSoup objects
================================
.. code:: python
from markdownify import MarkdownConverter
# Create shorthand method for conversion
def md(soup, **options):
return MarkdownConverter(**options).convert_soup(soup)
Creating Custom Converters Creating Custom Converters
========================== ==========================

View File

@@ -25,14 +25,6 @@ ASTERISK = '*'
UNDERSCORE = '_' UNDERSCORE = '_'
def escape(text, escape_underscores):
if not text:
return ''
if escape_underscores:
return text.replace('_', r'\_')
return text
def chomp(text): def chomp(text):
""" """
If the text in an inline tag like b, a, or em contains a leading or trailing If the text in an inline tag like b, a, or em contains a leading or trailing
@@ -71,10 +63,13 @@ class MarkdownConverter(object):
autolinks = True autolinks = True
bullets = '*+-' # An iterable of bullet types. bullets = '*+-' # An iterable of bullet types.
code_language = '' code_language = ''
code_language_callback = None
convert = None convert = None
default_title = False default_title = False
escape_asterisks = True
escape_underscores = True escape_underscores = True
heading_style = UNDERLINED heading_style = UNDERLINED
keep_inline_images_in = []
newline_style = SPACES newline_style = SPACES
strip = None strip = None
strong_em_symbol = ASTERISK strong_em_symbol = ASTERISK
@@ -96,6 +91,9 @@ class MarkdownConverter(object):
def convert(self, html): def convert(self, html):
soup = BeautifulSoup(html, 'html.parser') soup = BeautifulSoup(html, 'html.parser')
return self.convert_soup(soup)
def convert_soup(self, soup):
return self.process_tag(soup, convert_as_inline=False, children_only=True) return self.process_tag(soup, convert_as_inline=False, children_only=True)
def process_tag(self, node, convert_as_inline, children_only=False): def process_tag(self, node, convert_as_inline, children_only=False):
@@ -158,7 +156,7 @@ class MarkdownConverter(object):
text = whitespace_re.sub(' ', text) text = whitespace_re.sub(' ', text)
if el.parent.name != 'code': if el.parent.name != 'code':
text = escape(text, self.options['escape_underscores']) text = self.escape(text)
# remove trailing whitespaces if any of the following condition is true: # remove trailing whitespaces if any of the following condition is true:
# - current text node is the last node in li # - current text node is the last node in li
@@ -196,6 +194,15 @@ class MarkdownConverter(object):
else: else:
return True return True
def escape(self, text):
if not text:
return ''
if self.options['escape_asterisks']:
text = text.replace('*', r'\*')
if self.options['escape_underscores']:
text = text.replace('_', r'\_')
return text
def indent(self, text, level): def indent(self, text, level):
return line_beginning_re.sub('\t' * level, text) if text else '' return line_beginning_re.sub('\t' * level, text) if text else ''
@@ -275,7 +282,8 @@ class MarkdownConverter(object):
src = el.attrs.get('src', None) or '' src = el.attrs.get('src', None) or ''
title = el.attrs.get('title', None) or '' title = el.attrs.get('title', None) or ''
title_part = ' "%s"' % title.replace('"', r'\"') if title else '' title_part = ' "%s"' % title.replace('"', r'\"') if title else ''
if convert_as_inline: if (convert_as_inline
and el.parent.name not in self.options['keep_inline_images_in']):
return alt return alt
return '![%s](%s%s)' % (alt, src, title_part) return '![%s](%s%s)' % (alt, src, title_part)
@@ -328,7 +336,12 @@ class MarkdownConverter(object):
def convert_pre(self, el, text, convert_as_inline): def convert_pre(self, el, text, convert_as_inline):
if not text: if not text:
return '' return ''
return '\n```%s\n%s\n```\n' % (self.options['code_language'], text) code_language = self.options['code_language']
if self.options['code_language_callback']:
code_language = self.options['code_language_callback'](el) or code_language
return '\n```%s\n%s\n```\n' % (code_language, text)
convert_s = convert_del convert_s = convert_del

View File

@@ -10,7 +10,7 @@ read = lambda filepath: codecs.open(filepath, 'r', 'utf-8').read()
pkgmeta = { pkgmeta = {
'__title__': 'markdownify', '__title__': 'markdownify',
'__author__': 'Matthew Tretter', '__author__': 'Matthew Tretter',
'__version__': '0.10.2', '__version__': '0.11.0',
} }

View File

@@ -133,12 +133,13 @@ def test_hn_nested_simple_tag():
def test_hn_nested_img(): def test_hn_nested_img():
image_attributes_to_markdown = [ image_attributes_to_markdown = [
("", ""), ("", "", ""),
("alt='Alt Text'", "Alt Text"), ("alt='Alt Text'", "Alt Text", ""),
("alt='Alt Text' title='Optional title'", "Alt Text"), ("alt='Alt Text' title='Optional title'", "Alt Text", " \"Optional title\""),
] ]
for image_attributes, markdown in image_attributes_to_markdown: for image_attributes, markdown, title in image_attributes_to_markdown:
assert md('<h3>A <img src="/path/to/img.jpg " ' + image_attributes + '/> B</h3>') == '### A ' + markdown + ' B\n\n' assert md('<h3>A <img src="/path/to/img.jpg" ' + image_attributes + '/> B</h3>') == '### A ' + markdown + ' B\n\n'
assert md('<h3>A <img src="/path/to/img.jpg" ' + image_attributes + '/> B</h3>', keep_inline_images_in=['h3']) == '### A ![' + markdown + '](/path/to/img.jpg' + title + ') B\n\n'
def test_hn_atx_headings(): def test_hn_atx_headings():
@@ -215,3 +216,12 @@ def test_sup():
def test_lang(): def test_lang():
assert md('<pre>test\n foo\nbar</pre>', code_language='python') == '\n```python\ntest\n foo\nbar\n```\n' assert md('<pre>test\n foo\nbar</pre>', code_language='python') == '\n```python\ntest\n foo\nbar\n```\n'
assert md('<pre><code>test\n foo\nbar</code></pre>', code_language='javascript') == '\n```javascript\ntest\n foo\nbar\n```\n' assert md('<pre><code>test\n foo\nbar</code></pre>', code_language='javascript') == '\n```javascript\ntest\n foo\nbar\n```\n'
def test_lang_callback():
def callback(el):
return el['class'][0] if el.has_attr('class') else None
assert md('<pre class="python">test\n foo\nbar</pre>', code_language_callback=callback) == '\n```python\ntest\n foo\nbar\n```\n'
assert md('<pre class="javascript"><code>test\n foo\nbar</code></pre>', code_language_callback=callback) == '\n```javascript\ntest\n foo\nbar\n```\n'
assert md('<pre class="javascript"><code class="javascript">test\n foo\nbar</code></pre>', code_language_callback=callback) == '\n```javascript\ntest\n foo\nbar\n```\n'

View File

@@ -1,4 +1,5 @@
from markdownify import MarkdownConverter from markdownify import MarkdownConverter
from bs4 import BeautifulSoup
class ImageBlockConverter(MarkdownConverter): class ImageBlockConverter(MarkdownConverter):
@@ -16,3 +17,9 @@ def test_img():
assert md('<img src="/path/to/img.jpg" alt="Alt text" title="Optional title" />') == '![Alt text](/path/to/img.jpg "Optional title")\n\n' assert md('<img src="/path/to/img.jpg" alt="Alt text" title="Optional title" />') == '![Alt text](/path/to/img.jpg "Optional title")\n\n'
assert md('<img src="/path/to/img.jpg" alt="Alt text" />') == '![Alt text](/path/to/img.jpg)\n\n' assert md('<img src="/path/to/img.jpg" alt="Alt text" />') == '![Alt text](/path/to/img.jpg)\n\n'
def test_soup():
html = '<b>test</b>'
soup = BeautifulSoup(html, 'html.parser')
assert MarkdownConverter().convert_soup(soup) == '**test**'

View File

@@ -1,6 +1,11 @@
from markdownify import markdownify as md from markdownify import markdownify as md
def test_asterisks():
assert md('*hey*dude*') == r'\*hey\*dude\*'
assert md('*hey*dude*', escape_asterisks=False) == r'*hey*dude*'
def test_underscore(): def test_underscore():
assert md('_hey_dude_') == r'\_hey\_dude\_' assert md('_hey_dude_') == r'\_hey\_dude\_'
assert md('_hey_dude_', escape_underscores=False) == r'_hey_dude_' assert md('_hey_dude_', escape_underscores=False) == r'_hey_dude_'