부록 #2. AI 응답 markdown 포맷 변환¶
변경 파일을 한 번에 덮어쓰기 하실려면, pyhub-git-commit-apply 유틸리티 설치하신 후에, 프로젝트 루트에서 아래 명령 실행하시면 지정 커밋의 모든 파일을 다운받아 현재 경로에 덮어쓰기합니다.
python -m pyhub_git_commit_apply https://github.com/pyhub-kr/django-webchat-rag-langcon2025/commit/34e8d36b6aa5e94ead0223c8b9181d933ce9291c
uv
를 사용하실 경우
uv run pyhub-git-commit-apply https://github.com/pyhub-kr/django-webchat-rag-langcon2025/commit/34e8d36b6aa5e94ead0223c8b9181d933ce9291c
markdown to html 변환은 서버 단에서 해도 되고, 클라이언트 단에서 해도 됩니다. 서버 단에서 수행하면 보다 풍부한 포맷 변환이 가능하지만, markdown 변환은 클라이언트 단에서 수행해도 충분할 듯 보입니다.
여러 라이브러리가 있지만 Showdown 라이브러리를 적용해보겠습니다.

chat/templates/chat/base.html
파일 덮어쓰기
1{% load static %}
2
3<!doctype html>
4<html>
5<head>
6 <meta charset="UTF-8">
7 <meta name="viewport" content="width=device-width, initial-scale=1.0">
8 <title>{% block title %}Django Chat{% endblock %}</title>
9 {# https://daisyui.com/docs/cdn/ #}
10 <link href="https://cdn.jsdelivr.net/npm/daisyui@4.12.24/dist/full.min.css" rel="stylesheet" type="text/css" />
11 <script src="https://cdn.tailwindcss.com"></script>
12 <script src="https://unpkg.com/htmx.org"></script>
13 <script src="https://unpkg.com/alpinejs"></script>
14 <script src="{% static 'rag/showdown/2.1.0/showdown.js' %}"></script>
15 <script src="{% static 'rag/markdown.js' %}"></script>
16</head>
17<body class="bg-gray-100">
18 <div class="container mx-auto px-4 py-8">
19 <header class="mb-8">
20 <nav class="bg-white shadow-lg rounded-lg">
21 <div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
22 <div class="flex justify-between h-16">
23 <div class="flex">
24 <div class="flex-shrink-0 flex items-center">
25 <a href="{% url 'chat:room_list' %}" class="text-xl font-bold text-gray-800">
26 Django Chat
27 </a>
28 </div>
29 </div>
30 <div class="flex items-center">
31 <a href="{% url 'chat:room_new' %}"
32 class="inline-flex items-center px-4 py-2 border border-transparent text-sm font-medium rounded-md text-white bg-indigo-600 hover:bg-indigo-700">
33 새 채팅방
34 </a>
35 </div>
36 </div>
37 </div>
38 </nav>
39 </header>
40
41 <main class="bg-white shadow-lg rounded-lg p-6">
42 {% block content %}
43 {% endblock %}
44 </main>
45
46 <footer class="mt-8 text-center text-gray-600 text-sm">
47 <p>© 2025 파이썬사랑방. All rights reserved.</p>
48 </footer>
49 </div>
50</body>
51</html>
chat/templates/chat/_message_list.html
파일 덮어쓰기
1{% load rag_tags %}
2
3{% for message in message_list %}
4 {% if message.role == "user" %}
5 <div class="chat chat-start">
6 <div class="chat-bubble">
7 {{ message.content }}
8 </div>
9 </div>
10 {% else %}
11 {# uuid4 포맷의 랜덤 id 발행 #}
12 {% uuid4_id as message_id %}
13 {# 지정 id로 메시지 문자열을 json 변환 #}
14 {{ message.content|json_script:message_id }}
15 <div class="chat chat-end">
16 <div class="chat-bubble"
17 x-data
18 x-init="
19 const jsonString = JSON.parse(document.getElementById('{{ message_id }}').textContent);
20 $el.innerHTML = markdownToHtml(jsonString);
21 ">
22 </div>
23 </div>
24 {% endif %}
25{% endfor %}