Coverage for promplate/prompt/builder.py: 90%
29 statements
« prev ^ index » next coverage.py v7.6.9, created at 2024-12-18 19:52 +0800
« prev ^ index » next coverage.py v7.6.9, created at 2024-12-18 19:52 +0800
1from types import FunctionType
4class CodeBuilder:
5 """Build source code conveniently."""
7 def __init__(self, indent_level=0, indent_str="\t"):
8 self.code = []
9 self.indent_level = indent_level
10 self.indent_str = indent_str
12 def __str__(self):
13 return "".join(map(str, self.code))
15 def add_line(self, line):
16 """Add a line of source to the code."""
17 self.code.extend((self.indent_str * self.indent_level, line, "\n"))
19 return self
21 def add_section(self):
22 """Add a section, a sub-CodeBuilder."""
23 section = CodeBuilder(self.indent_level, self.indent_str)
24 self.code.append(section)
26 return section
28 def indent(self):
29 """Increase the current indent for following lines."""
30 self.indent_level += 1
32 return self
34 def dedent(self):
35 """Decrease the current indent for following lines."""
36 self.indent_level -= 1
38 return self
40 def get_render_function(self) -> FunctionType:
41 """Execute the code, and return a dict of globals it defines."""
42 assert self.indent_level == 0
43 python_source = str(self)
44 global_namespace = {}
45 exec(python_source, global_namespace)
46 return global_namespace["render"]
49def get_base_builder(sync=True, indent_str="\t"):
50 return (
51 CodeBuilder(indent_str=indent_str)
52 .add_line("def render():" if sync else "async def render():")
53 .indent()
54 .add_line("__parts__ = []")
55 .add_line("__append__ = __parts__.append")
56 )