mirror of
https://github.com/ggml-org/llama.cpp.git
synced 2025-10-27 08:21:30 +00:00
* feat: add Jinja tester PySide6 simple app * Linter fixes * Pylint fixes * Whitespace * Add commandline support; add formatter; add extensions * Remove testing actions * Silence flake8 warnings for commandline mode * Apply suggestions from code review Co-authored-by: Sigbjørn Skjæret <sigbjorn.skjaeret@scala.com> * Fix trailing whitespace/newline logic * Update scripts/jinja/jinja-tester.py Co-authored-by: Sigbjørn Skjæret <sigbjorn.skjaeret@scala.com> * Update scripts/jinja/jinja-tester.py Co-authored-by: Sigbjørn Skjæret <sigbjorn.skjaeret@scala.com> --------- Co-authored-by: Sigbjørn Skjæret <sigbjorn.skjaeret@scala.com>
505 lines
18 KiB
Python
Executable File
505 lines
18 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
import sys
|
|
import json
|
|
import argparse
|
|
import jinja2.ext as jinja2_ext
|
|
from PySide6.QtWidgets import (
|
|
QApplication,
|
|
QMainWindow,
|
|
QWidget,
|
|
QVBoxLayout,
|
|
QHBoxLayout,
|
|
QLabel,
|
|
QPlainTextEdit,
|
|
QTextEdit,
|
|
QPushButton,
|
|
QFileDialog,
|
|
)
|
|
from PySide6.QtGui import QColor, QColorConstants, QTextCursor, QTextFormat
|
|
from PySide6.QtCore import Qt, QRect, QSize
|
|
from jinja2 import TemplateSyntaxError
|
|
from jinja2.sandbox import ImmutableSandboxedEnvironment
|
|
from datetime import datetime
|
|
|
|
|
|
def format_template_content(template_content):
|
|
"""Format the Jinja template content using Jinja2's lexer."""
|
|
if not template_content.strip():
|
|
return template_content
|
|
|
|
env = ImmutableSandboxedEnvironment()
|
|
tc_rstrip = template_content.rstrip()
|
|
tokens = list(env.lex(tc_rstrip))
|
|
result = ""
|
|
indent_level = 0
|
|
i = 0
|
|
|
|
while i < len(tokens):
|
|
token = tokens[i]
|
|
_, token_type, token_value = token
|
|
|
|
if token_type == "block_begin":
|
|
block_start = i
|
|
# Collect all tokens for this block construct
|
|
construct_content = token_value
|
|
end_token_type = token_type.replace("_begin", "_end")
|
|
j = i + 1
|
|
while j < len(tokens) and tokens[j][1] != end_token_type:
|
|
construct_content += tokens[j][2]
|
|
j += 1
|
|
|
|
if j < len(tokens): # Found the end token
|
|
construct_content += tokens[j][2]
|
|
i = j # Skip to the end token
|
|
|
|
# Check for control structure keywords for indentation
|
|
stripped_content = construct_content.strip()
|
|
instr = block_start + 1
|
|
while tokens[instr][1] == "whitespace":
|
|
instr = instr + 1
|
|
|
|
instruction_token = tokens[instr][2]
|
|
start_control_tokens = ["if", "for", "macro", "call", "block"]
|
|
end_control_tokens = ["end" + t for t in start_control_tokens]
|
|
is_control_start = any(
|
|
instruction_token.startswith(kw) for kw in start_control_tokens
|
|
)
|
|
is_control_end = any(
|
|
instruction_token.startswith(kw) for kw in end_control_tokens
|
|
)
|
|
|
|
# Adjust indentation for control structures
|
|
# For control end blocks, decrease indent BEFORE adding the content
|
|
if is_control_end:
|
|
indent_level = max(0, indent_level - 1)
|
|
|
|
# Remove all previous whitespace before this block
|
|
result = result.rstrip()
|
|
|
|
# Add proper indent, but only if this is not the first token
|
|
added_newline = False
|
|
if result: # Only add newline and indent if there's already content
|
|
result += (
|
|
"\n" + " " * indent_level
|
|
) # Use 2 spaces per indent level
|
|
added_newline = True
|
|
else: # For the first token, don't add any indent
|
|
result += ""
|
|
|
|
# Add the block content
|
|
result += stripped_content
|
|
|
|
# Add '-' after '%' if it wasn't there and we added a newline or indent
|
|
if (
|
|
added_newline
|
|
and stripped_content.startswith("{%")
|
|
and not stripped_content.startswith("{%-")
|
|
):
|
|
# Add '-' at the beginning
|
|
result = (
|
|
result[: result.rfind("{%")]
|
|
+ "{%-"
|
|
+ result[result.rfind("{%") + 2 :]
|
|
)
|
|
if stripped_content.endswith("%}") and not stripped_content.endswith(
|
|
"-%}"
|
|
):
|
|
# Only add '-' if this is not the last token or if there's content after
|
|
if i + 1 < len(tokens) and tokens[i + 1][1] != "eof":
|
|
result = result[:-2] + "-%}"
|
|
|
|
# For control start blocks, increase indent AFTER adding the content
|
|
if is_control_start:
|
|
indent_level += 1
|
|
else:
|
|
# Malformed template, just add the token
|
|
result += token_value
|
|
elif token_type == "variable_begin":
|
|
# Collect all tokens for this variable construct
|
|
construct_content = token_value
|
|
end_token_type = token_type.replace("_begin", "_end")
|
|
j = i + 1
|
|
while j < len(tokens) and tokens[j][1] != end_token_type:
|
|
construct_content += tokens[j][2]
|
|
j += 1
|
|
|
|
if j < len(tokens): # Found the end token
|
|
construct_content += tokens[j][2]
|
|
i = j # Skip to the end token
|
|
|
|
# For variable constructs, leave them alone
|
|
# Do not add indent or whitespace before or after them
|
|
result += construct_content
|
|
else:
|
|
# Malformed template, just add the token
|
|
result += token_value
|
|
elif token_type == "data":
|
|
# Handle data (text between Jinja constructs)
|
|
# For data content, preserve it as is
|
|
result += token_value
|
|
else:
|
|
# Handle any other tokens
|
|
result += token_value
|
|
|
|
i += 1
|
|
|
|
# Clean up trailing newlines and spaces
|
|
result = result.rstrip()
|
|
|
|
# Copy the newline / space count from the original
|
|
if (trailing_length := len(template_content) - len(tc_rstrip)):
|
|
result += template_content[-trailing_length:]
|
|
|
|
return result
|
|
|
|
|
|
# ------------------------
|
|
# Line Number Widget
|
|
# ------------------------
|
|
class LineNumberArea(QWidget):
|
|
def __init__(self, editor):
|
|
super().__init__(editor)
|
|
self.code_editor = editor
|
|
|
|
def sizeHint(self):
|
|
return QSize(self.code_editor.line_number_area_width(), 0)
|
|
|
|
def paintEvent(self, event):
|
|
self.code_editor.line_number_area_paint_event(event)
|
|
|
|
|
|
class CodeEditor(QPlainTextEdit):
|
|
def __init__(self):
|
|
super().__init__()
|
|
self.line_number_area = LineNumberArea(self)
|
|
|
|
self.blockCountChanged.connect(self.update_line_number_area_width)
|
|
self.updateRequest.connect(self.update_line_number_area)
|
|
self.cursorPositionChanged.connect(self.highlight_current_line)
|
|
|
|
self.update_line_number_area_width(0)
|
|
self.highlight_current_line()
|
|
|
|
def line_number_area_width(self):
|
|
digits = len(str(self.blockCount()))
|
|
space = 3 + self.fontMetrics().horizontalAdvance("9") * digits
|
|
return space
|
|
|
|
def update_line_number_area_width(self, _):
|
|
self.setViewportMargins(self.line_number_area_width(), 0, 0, 0)
|
|
|
|
def update_line_number_area(self, rect, dy):
|
|
if dy:
|
|
self.line_number_area.scroll(0, dy)
|
|
else:
|
|
self.line_number_area.update(
|
|
0, rect.y(), self.line_number_area.width(), rect.height()
|
|
)
|
|
|
|
if rect.contains(self.viewport().rect()):
|
|
self.update_line_number_area_width(0)
|
|
|
|
def resizeEvent(self, event):
|
|
super().resizeEvent(event)
|
|
cr = self.contentsRect()
|
|
self.line_number_area.setGeometry(
|
|
QRect(cr.left(), cr.top(), self.line_number_area_width(), cr.height())
|
|
)
|
|
|
|
def line_number_area_paint_event(self, event):
|
|
from PySide6.QtGui import QPainter
|
|
|
|
painter = QPainter(self.line_number_area)
|
|
painter.fillRect(event.rect(), QColorConstants.LightGray)
|
|
|
|
block = self.firstVisibleBlock()
|
|
block_number = block.blockNumber()
|
|
top = int(
|
|
self.blockBoundingGeometry(block).translated(self.contentOffset()).top()
|
|
)
|
|
bottom = top + int(self.blockBoundingRect(block).height())
|
|
|
|
while block.isValid() and top <= event.rect().bottom():
|
|
if block.isVisible() and bottom >= event.rect().top():
|
|
number = str(block_number + 1)
|
|
painter.setPen(QColorConstants.Black)
|
|
painter.drawText(
|
|
0,
|
|
top,
|
|
self.line_number_area.width() - 2,
|
|
self.fontMetrics().height(),
|
|
Qt.AlignmentFlag.AlignRight,
|
|
number,
|
|
)
|
|
block = block.next()
|
|
top = bottom
|
|
bottom = top + int(self.blockBoundingRect(block).height())
|
|
block_number += 1
|
|
|
|
def highlight_current_line(self):
|
|
extra_selections = []
|
|
if not self.isReadOnly():
|
|
selection = QTextEdit.ExtraSelection()
|
|
line_color = QColorConstants.Yellow.lighter(160)
|
|
selection.format.setBackground(line_color) # pyright: ignore[reportAttributeAccessIssue]
|
|
selection.format.setProperty(QTextFormat.Property.FullWidthSelection, True) # pyright: ignore[reportAttributeAccessIssue]
|
|
selection.cursor = self.textCursor() # pyright: ignore[reportAttributeAccessIssue]
|
|
selection.cursor.clearSelection() # pyright: ignore[reportAttributeAccessIssue]
|
|
extra_selections.append(selection)
|
|
self.setExtraSelections(extra_selections)
|
|
|
|
def highlight_position(self, lineno: int, col: int, color: QColor):
|
|
block = self.document().findBlockByLineNumber(lineno - 1)
|
|
if block.isValid():
|
|
cursor = QTextCursor(block)
|
|
text = block.text()
|
|
start = block.position() + max(0, col - 1)
|
|
cursor.setPosition(start)
|
|
if col <= len(text):
|
|
cursor.movePosition(
|
|
QTextCursor.MoveOperation.NextCharacter,
|
|
QTextCursor.MoveMode.KeepAnchor,
|
|
)
|
|
|
|
extra = QTextEdit.ExtraSelection()
|
|
extra.format.setBackground(color.lighter(160)) # pyright: ignore[reportAttributeAccessIssue]
|
|
extra.cursor = cursor # pyright: ignore[reportAttributeAccessIssue]
|
|
|
|
self.setExtraSelections(self.extraSelections() + [extra])
|
|
|
|
def highlight_line(self, lineno: int, color: QColor):
|
|
block = self.document().findBlockByLineNumber(lineno - 1)
|
|
if block.isValid():
|
|
cursor = QTextCursor(block)
|
|
cursor.select(QTextCursor.SelectionType.LineUnderCursor)
|
|
|
|
extra = QTextEdit.ExtraSelection()
|
|
extra.format.setBackground(color.lighter(160)) # pyright: ignore[reportAttributeAccessIssue]
|
|
extra.cursor = cursor # pyright: ignore[reportAttributeAccessIssue]
|
|
|
|
self.setExtraSelections(self.extraSelections() + [extra])
|
|
|
|
def clear_highlighting(self):
|
|
self.highlight_current_line()
|
|
|
|
|
|
# ------------------------
|
|
# Main App
|
|
# ------------------------
|
|
class JinjaTester(QMainWindow):
|
|
def __init__(self):
|
|
super().__init__()
|
|
self.setWindowTitle("Jinja Template Tester")
|
|
self.resize(1200, 800)
|
|
|
|
central = QWidget()
|
|
main_layout = QVBoxLayout(central)
|
|
|
|
# -------- Top input area --------
|
|
input_layout = QHBoxLayout()
|
|
|
|
# Template editor with label
|
|
template_layout = QVBoxLayout()
|
|
template_label = QLabel("Jinja2 Template")
|
|
template_layout.addWidget(template_label)
|
|
self.template_edit = CodeEditor()
|
|
template_layout.addWidget(self.template_edit)
|
|
input_layout.addLayout(template_layout)
|
|
|
|
# JSON editor with label
|
|
json_layout = QVBoxLayout()
|
|
json_label = QLabel("Context (JSON)")
|
|
json_layout.addWidget(json_label)
|
|
self.json_edit = CodeEditor()
|
|
self.json_edit.setPlainText("""
|
|
{
|
|
"add_generation_prompt": true,
|
|
"bos_token": "",
|
|
"eos_token": "",
|
|
"messages": [
|
|
{
|
|
"role": "user",
|
|
"content": "What is the capital of Poland?"
|
|
}
|
|
]
|
|
}
|
|
""".strip())
|
|
json_layout.addWidget(self.json_edit)
|
|
input_layout.addLayout(json_layout)
|
|
|
|
main_layout.addLayout(input_layout)
|
|
|
|
# -------- Rendered output area --------
|
|
output_label = QLabel("Rendered Output")
|
|
main_layout.addWidget(output_label)
|
|
self.output_edit = QPlainTextEdit()
|
|
self.output_edit.setReadOnly(True)
|
|
main_layout.addWidget(self.output_edit)
|
|
|
|
# -------- Render button and status --------
|
|
btn_layout = QHBoxLayout()
|
|
|
|
# Load template button
|
|
self.load_btn = QPushButton("Load Template")
|
|
self.load_btn.clicked.connect(self.load_template)
|
|
btn_layout.addWidget(self.load_btn)
|
|
|
|
# Format template button
|
|
self.format_btn = QPushButton("Format")
|
|
self.format_btn.clicked.connect(self.format_template)
|
|
btn_layout.addWidget(self.format_btn)
|
|
|
|
self.render_btn = QPushButton("Render")
|
|
self.render_btn.clicked.connect(self.render_template)
|
|
btn_layout.addWidget(self.render_btn)
|
|
main_layout.addLayout(btn_layout)
|
|
|
|
# Status label below buttons
|
|
self.status_label = QLabel("Ready")
|
|
main_layout.addWidget(self.status_label)
|
|
|
|
self.setCentralWidget(central)
|
|
|
|
def render_template(self):
|
|
self.template_edit.clear_highlighting()
|
|
self.output_edit.clear()
|
|
|
|
template_str = self.template_edit.toPlainText()
|
|
json_str = self.json_edit.toPlainText()
|
|
|
|
# Parse JSON context
|
|
try:
|
|
context = json.loads(json_str) if json_str.strip() else {}
|
|
except Exception as e:
|
|
self.status_label.setText(f"❌ JSON Error: {e}")
|
|
return
|
|
|
|
def raise_exception(text: str) -> str:
|
|
raise RuntimeError(text)
|
|
|
|
env = ImmutableSandboxedEnvironment(
|
|
trim_blocks=True,
|
|
lstrip_blocks=True,
|
|
extensions=[jinja2_ext.loopcontrols],
|
|
)
|
|
env.filters["tojson"] = (
|
|
lambda x,
|
|
indent=None,
|
|
separators=None,
|
|
sort_keys=False,
|
|
ensure_ascii=False: json.dumps(
|
|
x,
|
|
indent=indent,
|
|
separators=separators,
|
|
sort_keys=sort_keys,
|
|
ensure_ascii=ensure_ascii,
|
|
)
|
|
)
|
|
env.globals["strftime_now"] = lambda format: datetime.now().strftime(format)
|
|
env.globals["raise_exception"] = raise_exception
|
|
try:
|
|
template = env.from_string(template_str)
|
|
output = template.render(context)
|
|
self.output_edit.setPlainText(output)
|
|
self.status_label.setText("✅ Render successful")
|
|
except TemplateSyntaxError as e:
|
|
self.status_label.setText(f"❌ Syntax Error (line {e.lineno}): {e.message}")
|
|
if e.lineno:
|
|
self.template_edit.highlight_line(e.lineno, QColor("red"))
|
|
except Exception as e:
|
|
# Catch all runtime errors
|
|
# Try to extract template line number
|
|
lineno = None
|
|
tb = e.__traceback__
|
|
while tb:
|
|
frame = tb.tb_frame
|
|
if frame.f_code.co_filename == "<template>":
|
|
lineno = tb.tb_lineno
|
|
break
|
|
tb = tb.tb_next
|
|
|
|
error_msg = f"Runtime Error: {type(e).__name__}: {e}"
|
|
if lineno:
|
|
error_msg = f"Runtime Error at line {lineno} in template: {type(e).__name__}: {e}"
|
|
self.template_edit.highlight_line(lineno, QColor("orange"))
|
|
|
|
self.output_edit.setPlainText(error_msg)
|
|
self.status_label.setText(f"❌ {error_msg}")
|
|
|
|
def load_template(self):
|
|
"""Load a Jinja template from a file using a file dialog."""
|
|
file_path, _ = QFileDialog.getOpenFileName(
|
|
self,
|
|
"Load Jinja Template",
|
|
"",
|
|
"Template Files (*.jinja *.j2 *.html *.txt);;All Files (*)",
|
|
)
|
|
|
|
if file_path:
|
|
try:
|
|
with open(file_path, "r", encoding="utf-8") as file:
|
|
content = file.read()
|
|
self.template_edit.setPlainText(content)
|
|
self.status_label.setText(f"✅ Loaded template from {file_path}")
|
|
except Exception as e:
|
|
self.status_label.setText(f"❌ Error loading file: {str(e)}")
|
|
|
|
def format_template(self):
|
|
"""Format the Jinja template using Jinja2's lexer for proper parsing."""
|
|
try:
|
|
template_content = self.template_edit.toPlainText()
|
|
if not template_content.strip():
|
|
self.status_label.setText("⚠️ Template is empty")
|
|
return
|
|
|
|
formatted_content = format_template_content(template_content)
|
|
self.template_edit.setPlainText(formatted_content)
|
|
self.status_label.setText("✅ Template formatted")
|
|
except Exception as e:
|
|
self.status_label.setText(f"❌ Error formatting template: {str(e)}")
|
|
|
|
|
|
if __name__ == "__main__":
|
|
if len(sys.argv) > 1:
|
|
# CLI mode
|
|
parser = argparse.ArgumentParser(description="Jinja Template Tester")
|
|
parser.add_argument(
|
|
"--template", required=True, help="Path to Jinja template file"
|
|
)
|
|
parser.add_argument("--context", required=True, help="JSON string for context")
|
|
parser.add_argument(
|
|
"--action",
|
|
choices=["format", "render"],
|
|
default="render",
|
|
help="Action to perform",
|
|
)
|
|
args = parser.parse_args()
|
|
|
|
# Load template
|
|
with open(args.template, "r", encoding="utf-8") as f:
|
|
template_content = f.read()
|
|
|
|
# Load JSON
|
|
context = json.loads(args.context)
|
|
# Add missing variables
|
|
context.setdefault("bos_token", "")
|
|
context.setdefault("eos_token", "")
|
|
context.setdefault("add_generation_prompt", False)
|
|
|
|
env = ImmutableSandboxedEnvironment()
|
|
|
|
if args.action == "format":
|
|
formatted = format_template_content(template_content)
|
|
print(formatted) # noqa: NP100
|
|
elif args.action == "render":
|
|
template = env.from_string(template_content)
|
|
output = template.render(context)
|
|
print(output) # noqa: NP100
|
|
|
|
else:
|
|
# GUI mode
|
|
app = QApplication(sys.argv)
|
|
window = JinjaTester()
|
|
window.show()
|
|
sys.exit(app.exec())
|