The following is all generated from within the python code.
in this blog post we are going to walk through the process of creating a markdown generator of python heavy code using python. The goal is to allow you to annotate your python code with markers and have it automatically generate markdown in the specified way. I was trying to write a different blog post and I realized that I couldn't find the tool I wanted to generate my blog post from within python code. So I decided to write it myself.
I then realized that this tool was far more interesting than the blog post I was trying to write. So here we are. Writing a blog post about how to write a blog post generator from within said blog post generator. (Very meta :))
for example the above statement looked like this in my python code.
# this indicates that the following string should be rendered as markdown
markdown_string
"""
in this blog post we are going to walk through the process of creating a markdown generator of python heavy code
using python. The goal is to allow you to annotate your python code with markers and have it automatically
generate markdown in the specified way. I was trying to write a different blog post and I realized that I couldn't
find the tool I wanted to generate my blog post from within python code. So I decided to write it myself.
I then realized that this tool was far more interesting than the blog post I was trying to write. So here we are.
Writing a blog post about how to write a blog post generator from within said blog post generator. (Very meta :))
"""
And this entire section looked like this:
# This starts a code block tagged with the name "babies first code block".
# If you did not use a tag, it would add it to the current section.
# I didn't do that here since I want to print the code block after the following markdown string
code_section("babies first code block", skip_blog_stuff=False)
# this indicates that the following string should be rendered as markdown
markdown_string
"""
in this blog post we are going to walk through the process of creating a markdown generator of python heavy code
using python. The goal is to allow you to annotate your python code with markers and have it automatically
generate markdown in the specified way. I was trying to write a different blog post and I realized that I couldn't
find the tool I wanted to generate my blog post from within python code. So I decided to write it myself.
I then realized that this tool was far more interesting than the blog post I was trying to write. So here we are.
Writing a blog post about how to write a blog post generator from within said blog post generator. (Very meta :))
"""
end_code("babies first code block")
markdown_string
"""
for example the above statement looked like this in my python code.
"""
# this writes the previous code block to the current section
write_section("babies first code block")
markdown_string
"""
And this entire section looked like this:
"""
The first challenge is how do we create flags that we can use to indicate that a section of code should be rendered as markdown. The first idea that comes to mind is to use comment flags for example # markdown_start and # markdown_end and # code_start and # code_end. This would work but it would be a bit ugly and you wouldn't be able to easily pass arguments to the flags easily. Then it hit me, if I created dummy functions that took the arguments I wanted to pass it would be possible to use actual python code as flags and get all of python's auto complete functionality. We do this in the following fashion. Note: this is in another file.
from __future__ import annotations
from typing import *
def write_section(section_name: str, target_section_name: str = ''):
pass
def markdown_string(tag: str = '', keep_indent: bool = False, header_level: int = 0,table_of_contents: bool = False):
pass
def set_current_section(section_name: str):
pass
def code_section(section_name: str = '', keep_indent: bool = False, skip_comments: bool = False,skip_blog_stuff=True):
pass
def end_code(inner_section_name: str = ''):
pass
These functions do not do anything. They are just dummy functions that the user can use as flags. Since we don't want these flags to affect the code we are writing. These flags only become active when we are generating the markdown (which for the sake of your sanity should be done in a different file).
The next challenge is actually reading the code.
with (open(source_file, 'r') as f):
lines = f.readlines()
for line_number, line in enumerate(lines):
We now strip the indent from the line and check if it starts with one of our functions. If it does we add exec_ to the beginning and then execute it.
stripped_line = line.strip()
for function_name in ["write_section", "markdown_string", "code_section", "set_current_section"]:
if stripped_line.startswith(function_name):
exec('exec_' + self.parse_line(line))
exec is a built in python function that executes a string as python code. This is mind blindingly powerful I wouldn't recommend using it in production code but for this use case it is perfect.
What this means for us is that we don't need to write a parser for our flags. We can just let python itself do the parsing for us. This is a huge win because parsing is surprisingly hard.
Because exec works on the current scope we define the functions inside the loop. This means that these functions have access to all current variables.
Here's a code dump of the functions we defined. Don't worry about understanding them right now. We will go through them in detail later.