๐Ÿ’ฌ ์ฑ„ํŒ…๋ฐฉ/๋ฉ”์‹œ์ง€ ๋ชจ๋ธ ๋ฐ ๊ธฐ๋ณธ ํŽ˜์ด์ง€ ๊ตฌ์„ฑยถ

๋ชจ๋ธยถ

์ฑ„ํŒ…๋ฐฉ์„ ์ €์žฅํ•  ๋ชจ๋ธ๋กœ์„œ Room ๋ชจ๋ธ์„ ์ •์˜ํ•ฉ๋‹ˆ๋‹ค. .create_ai_message() ๋ฉ”์„œ๋“œ๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด ์ฑ„ํŒ…๋ฐฉ์˜ ์ด์ „ ๋ฉ”์‹œ์ง€๋“ค์„ ์ˆ˜์ง‘ํ•˜์—ฌ AI ์‘๋‹ต์„ ์ƒ์„ฑํ•˜๊ณ , ์ƒˆ AI ๋ฉ”์‹œ์ง€๋กœ์„œ ์ €์žฅํ•ฉ๋‹ˆ๋‹ค. ์ฑ„ํŒ…๋ฐฉ์˜ ์‹œ์Šคํ…œ ํ”„๋กฌํ”„ํŠธ๊ฐ€ ๋ณ€๊ฒฝ๋˜๋ฉด ์ด์ „ ๋ฉ”์‹œ์ง€๋“ค์„ ๋ชจ๋‘ ์‚ญ์ œํ•˜๋„๋ก ํ–ˆ์Šต๋‹ˆ๋‹ค. ๋ฉ”์‹œ์ง€ ์‚ญ์ œ๋Š” ์ƒํ™ฉ์— ๋”ฐ๋ผ ์ ์šฉํ•˜์ง€ ์•Š์•„๋„ ๋ฉ๋‹ˆ๋‹ค.

  • name: ์ฑ„ํŒ…๋ฐฉ ์ด๋ฆ„

  • system_prompt: ์ฑ„ํŒ…๋ฐฉ ์‹œ์Šคํ…œ ํ”„๋กฌํ”„ํŠธ

  • created_at: ์ฑ„ํŒ…๋ฐฉ ์ƒ์„ฑ์ผ์‹œ

  • updated_at: ์ฑ„ํŒ…๋ฐฉ ์ˆ˜์ •์ผ์‹œ

์ฑ„ํŒ… ๋ฉ”์‹œ์ง€๋ฅผ ์ €์žฅํ•  ๋ชจ๋ธ๋กœ์„œ Message ๋ชจ๋ธ์„ ์ •์˜ํ•ฉ๋‹ˆ๋‹ค.

  • role: ๋ฉ”์‹œ์ง€ ์—ญํ•  (user, assistant)

  • content: ๋ฉ”์‹œ์ง€ ๋‚ด์šฉ

  • created_at: ๋ฉ”์‹œ์ง€ ์ƒ์„ฑ์ผ์‹œ

  • updated_at: ๋ฉ”์‹œ์ง€ ์ˆ˜์ •์ผ์‹œ

  • room: ์ฑ„ํŒ…๋ฐฉ (Room ๋ชจ๋ธ ์™ธ๋ž˜ํ‚ค)

        erDiagram
    Room ||--o{ Message : has
    Room {
        string name
        string system_prompt
        datetime created_at
        datetime updated_at
    }
    Message {
        string role
        string content
        datetime created_at
        datetime updated_at
        string room_id
    }
    
chat/models.py ๋ฎ์–ด์“ฐ๊ธฐยถ
 1import json
 2
 3from django.db import models
 4from django.utils.functional import cached_property
 5from django_lifecycle import LifecycleModelMixin, hook, AFTER_UPDATE
 6from pyhub.rag.fields.sqlite import SQLiteVectorField
 7from pyhub.rag.models.sqlite import SQLiteVectorDocument
 8
 9from chat.llm import LLM
10
11
12class TaxLawDocument(SQLiteVectorDocument):
13    embedding = SQLiteVectorField(
14        dimensions=3072,
15        editable=False,
16        embedding_model="text-embedding-3-large",
17    )
18
19    @cached_property
20    def page_content_obj(self):
21        return json.loads(self.page_content)
22
23
24class Room(LifecycleModelMixin, models.Model):
25    name = models.CharField(max_length=255)
26    system_prompt = models.TextField(blank=True)
27    created_at = models.DateTimeField(auto_now_add=True)
28    updated_at = models.DateTimeField(auto_now=True)
29
30    def __str__(self):
31        return self.name
32
33    @hook(AFTER_UPDATE, when="system_prompt", has_changed=True)
34    def on_after_update(self):
35        self.message_set.all().delete()
36
37    def create_ai_message(self):
38        # ํ˜„์žฌ ๋ฐฉ์˜ ์ด์ „ ๋ฉ”์‹œ์ง€๋“ค์„ ์ˆ˜์ง‘
39        message_qs = self.message_set.all()
40        messages = [{"role": msg.role, "content": msg.content} for msg in message_qs]
41
42        # AI ์‘๋‹ต ์ƒ์„ฑ
43        llm = LLM(
44            model="gpt-4o-mini",
45            temperature=1,
46            system_prompt=self.system_prompt,
47            initial_messages=messages,
48        )
49        ai_message = llm.make_reply()
50
51        # AI ์‘๋‹ต์„ ์ƒˆ ๋ฉ”์‹œ์ง€๋กœ ์ €์žฅ
52        return self.message_set.create(
53            role=Message.Role.ASSISTANT,
54            content=ai_message,
55        )
56
57    class Meta:
58        ordering = ["-pk"]
59
60
61class Message(models.Model):
62    class Role(models.TextChoices):
63        USER = "user"
64        ASSISTANT = "assistant"
65
66    room = models.ForeignKey(Room, on_delete=models.CASCADE)
67    role = models.CharField(max_length=255, choices=Role.choices, default=Role.USER)
68    content = models.TextField()
69    created_at = models.DateTimeField(auto_now_add=True)
70    updated_at = models.DateTimeField(auto_now=True)
71
72    def __str__(self):
73        return self.content
74
75    class Meta:
76        ordering = ["pk"]
chat/models.py ๋ฎ์–ด์“ฐ๊ธฐยถ
 1import json
 2
 3from django.db import models
 4from django.utils.functional import cached_property
 5from django_lifecycle import LifecycleModelMixin, hook, AFTER_UPDATE
 6from pyhub.rag.fields.postgres import PGVectorField
 7from pyhub.rag.models.postgres import PGVectorDocument
 8
 9from chat.llm import LLM
10
11
12class TaxLawDocument(PGVectorDocument):
13    embedding = PGVectorField(
14        dimensions=3072,
15        editable=False,
16        embedding_model="text-embedding-3-large",
17    )
18
19    @cached_property
20    def page_content_obj(self):
21        return json.loads(self.page_content)
22
23    class Meta:
24        indexes = [
25            PGVectorDocument.make_hnsw_index(
26                "chat_taxlawdoc_idx",
27                "halfvec",
28                "cosine",
29            ),
30        ]
31
32
33class Room(LifecycleModelMixin, models.Model):
34    name = models.CharField(max_length=255)
35    system_prompt = models.TextField(blank=True)
36    created_at = models.DateTimeField(auto_now_add=True)
37    updated_at = models.DateTimeField(auto_now=True)
38
39    def __str__(self):
40        return self.name
41
42    @hook(AFTER_UPDATE, when="system_prompt", has_changed=True)
43    def on_after_update(self):
44        self.message_set.all().delete()
45
46    def create_ai_message(self):
47        # ํ˜„์žฌ ๋ฐฉ์˜ ์ด์ „ ๋ฉ”์‹œ์ง€๋“ค์„ ์ˆ˜์ง‘
48        message_qs = self.message_set.all()
49        messages = [{"role": msg.role, "content": msg.content} for msg in message_qs]
50
51        # AI ์‘๋‹ต ์ƒ์„ฑ
52        llm = LLM(
53            model="gpt-4o-mini",
54            temperature=1,
55            system_prompt=self.system_prompt,
56            initial_messages=messages,
57        )
58        ai_message = llm.make_reply()
59
60        # AI ์‘๋‹ต์„ ์ƒˆ ๋ฉ”์‹œ์ง€๋กœ ์ €์žฅ
61        return self.message_set.create(
62            role=Message.Role.ASSISTANT,
63            content=ai_message,
64        )
65
66    class Meta:
67        ordering = ["-pk"]
68
69
70class Message(models.Model):
71    class Role(models.TextChoices):
72        USER = "user"
73        ASSISTANT = "assistant"
74
75    room = models.ForeignKey(Room, on_delete=models.CASCADE)
76    role = models.CharField(max_length=255, choices=Role.choices, default=Role.USER)
77    content = models.TextField()
78    created_at = models.DateTimeField(auto_now_add=True)
79    updated_at = models.DateTimeField(auto_now=True)
80
81    def __str__(self):
82        return self.content
83
84    class Meta:
85        ordering = ["pk"]

์ƒˆ๋กœ์šด ๋ชจ๋ธ์„ ์ •์˜ํ–ˆ์œผ๋‹ˆ, ๋งˆ์ด๊ทธ๋ ˆ์ด์…˜ ํŒŒ์ผ์„ ์ƒ์„ฑํ•˜๊ณ  (์ž‘์—… ์ง€์‹œ์„œ ์ƒ์„ฑ), ๋งˆ์ด๊ทธ๋ ˆ์ด์…˜์„ ํ†ตํ•ด ์ˆ˜ํ–‰๋  SQL ๋‚ด์—ญ์„ ํ™•์ธํ•˜๊ณ  (์ž‘์—… ์ง€์‹œ์„œ ํ™•์ธ), ๋งˆ์ด๊ทธ๋ ˆ์ด์…˜์„ ์ˆ˜ํ–‰ํ•ฉ๋‹ˆ๋‹ค (์ž‘์—… ์ง€์‹œ์„œ ์‹คํ–‰).

../../_images/0002-migrate.png

ํผยถ

์œ ์ €์—๊ฒŒ ์ฑ„ํŒ…๋ฐฉ๊ณผ ๋ฉ”์‹œ์ง€๋ฅผ ์ž…๋ ฅํ•  ์ˆ˜ ์žˆ๋Š” ํผ์„ ์ œ๊ณตํ•˜๊ณ , ์ž…๋ ฅ๊ฐ’์— ๋Œ€ํ•œ ์œ ํšจ์„ฑ ๊ฒ€์ฆ์„ ์œ„ํ•ด ๋ชจ๋ธํผ์„ ์ƒ์„ฑํ•ฉ๋‹ˆ๋‹ค.

chat/forms.py ํŒŒ์ผ ์ƒ์„ฑยถ
 1from django import forms
 2from .models import Message, Room
 3
 4
 5# ์ƒˆ ์ฑ„ํŒ…๋ฐฉ ์ƒ์„ฑ ๋ฐ ์ˆ˜์ • ํŽ˜์ด์ง€์—์„œ
 6# ์ž…๋ ฅ HTML ํผ ์ƒ์„ฑ ๋ฐ ์œ ํšจ์„ฑ ๊ฒ€์‚ฌ๋ฅผ ๋‹ด๋‹น
 7class RoomForm(forms.ModelForm):
 8    class Meta:
 9        model = Room
10        fields = ["name", "system_prompt"]
11
12
13# ์ฑ„ํŒ… ๋ฉ”์‹œ์ง€ ์ž…๋ ฅ/์ˆ˜์ • ํผ์„ ์ƒ์„ฑํ•˜๊ณ  ์œ ํšจ์„ฑ ๊ฒ€์‚ฌ๋ฅผ ๋‹ด๋‹น
14class MessageForm(forms.ModelForm):
15    class Meta:
16        model = Message
17        fields = ["content"]

๋ทฐยถ

3๊ฐœ์˜ ํŽ˜์ด์ง€๋ฅผ ๊ตฌํ˜„ํ•ฉ๋‹ˆ๋‹ค.

  • room_list: ์ฑ„ํŒ…๋ฐฉ ๋ชฉ๋ก ํŽ˜์ด์ง€

    • model = Room : ๋ชฉ๋ก์„ ๊ตฌ์„ฑํ•  ๋ชจ๋ธ

  • room_new: ์ƒˆ ์ฑ„ํŒ…๋ฐฉ ์ƒ์„ฑ ํŽ˜์ด์ง€

    • form = RoomForm : ์ž…๋ ฅ๊ฐ’์— ๋Œ€ํ•œ ์œ ํšจ์„ฑ ๊ฒ€์ฆ์„ ์ˆ˜ํ–‰ํ•˜๊ณ , ์ž…๋ ฅํผ HTML์„ ์ƒ์„ฑํ•  ํผ

    • room = Room : ์ž…๋ ฅ๊ฐ’์„ ์ €์žฅํ•  ๋ชจ๋ธ

    • success_url = reverse_lazy("chat:room_list") : ์ž…๋ ฅ๊ฐ’์„ ์ €์žฅํ•œ ํ›„ ์ด๋™ํ•  URL

  • room_detail: ์ฑ„ํŒ…๋ฐฉ ์ฑ„ํŒ… ํŽ˜์ด์ง€

    • ์ง€์ • pk ์˜ ์ฑ„ํŒ…๋ฐฉ์„ ์กฐํšŒํ•˜๊ณ , ํ•ด๋‹น ์ฑ„ํŒ…๋ฐฉ ๋‚ด ๋ชจ๋“  ๋ฉ”์‹œ์ง€๋ฅผ ์กฐํšŒํ•ฉ๋‹ˆ๋‹ค.

chat/views.py ํŒŒ์ผ ๋ฎ์–ด์“ฐ๊ธฐยถ
 1from django.shortcuts import get_object_or_404, render
 2from django.urls import reverse_lazy
 3from django.views.generic import ListView, CreateView
 4
 5from .forms import RoomForm
 6from .models import Room, TaxLawDocument
 7
 8
 9# ์ฑ„ํŒ…๋ฐฉ ๋ชฉ๋ก ํŽ˜์ด์ง€ (ํด๋ž˜์Šค ๊ธฐ๋ฐ˜ ๋ทฐ)
10room_list = ListView.as_view(model=Room)
11
12
13# ์ƒˆ ์ฑ„ํŒ…๋ฐฉ ์ƒ์„ฑ ํŽ˜์ด์ง€ (ํด๋ž˜์Šค ๊ธฐ๋ฐ˜ ๋ทฐ)
14room_new = CreateView.as_view(
15    model=Room,
16    form_class=RoomForm,
17    success_url=reverse_lazy("chat:room_list"),
18)
19
20
21# ์ฑ„ํŒ…๋ฐฉ ์ฑ„ํŒ… ํŽ˜์ด์ง€ (ํ•จ์ˆ˜ ๊ธฐ๋ฐ˜ ๋ทฐ)
22def room_detail(request, pk):
23    # ์ง€์ • ์ฑ„ํŒ…๋ฐฉ ์กฐํšŒํ•˜๊ณ , ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค์— ์—†์œผ๋ฉด 404 ์˜ค๋ฅ˜ ๋ฐœ์ƒ
24    room = get_object_or_404(Room, pk=pk)
25    # ์ง€์ • ์ฑ„ํŒ…๋ฐฉ์˜ ๋ชจ๋“  ๋Œ€ํ™” ๋ชฉ๋ก
26    message_list = room.message_set.all()
27    return render(
28        request,
29        "chat/room_detail.html",
30        {
31            "room": room,
32            "message_list": message_list,
33        },
34    )
35
36
37# ๋ฌธ์„œ ๊ฒ€์ƒ‰ ํŽ˜์ด์ง€
38class TaxLawDocumentListView(ListView):
39    model = TaxLawDocument
40    # sqlite์˜ similarity_search ๋ฉ”์„œ๋“œ๊ฐ€ ์ฟผ๋ฆฌ์…‹์ด ์•„๋‹Œ ๋ฆฌ์ŠคํŠธ๋ฅผ ๋ฐ˜ํ™˜ํ•˜๊ธฐ ๋•Œ๋ฌธ์—
41    # ListView์—์„œ ํ…œํ”Œ๋ฆฟ ์ด๋ฆ„์„ ์ฐพ์ง€ ๋ชปํ•˜๊ธฐ์— ์ง์ ‘ ์ง€์ •ํ•ด์ค๋‹ˆ๋‹ค.
42    template_name = "chat/taxlawdocument_list.html"
43
44    def get_queryset(self):
45        qs = super().get_queryset()
46
47        query = self.request.GET.get("query", "").strip()
48        if query:
49            qs = qs.similarity_search(query)  # noqa: list ํƒ€์ž…
50        else:
51            # ๊ฒ€์ƒ‰์–ด๊ฐ€ ์—†๋‹ค๋ฉด ๋นˆ ์ฟผ๋ฆฌ์…‹์„ ๋ฐ˜ํ™˜ํ•ฉ๋‹ˆ๋‹ค.
52            qs = qs.none()
53
54        return qs

๊ฐ ๋ทฐ์— ๋Œ€ํ•ด URL ํŒจํ„ด๋„ ์•ž์„œ ๐Ÿ” ๋ฌธ์„œ ๊ฒ€์ƒ‰ ํŽ˜์ด์ง€ ๋ฌธ์„œ์—์„œ ์ž‘์„ฑํ–ˆ์—ˆ์Šต๋‹ˆ๋‹ค.

ํ…œํ”Œ๋ฆฟยถ

chat/templates/chat/room_list.html ํŒŒ์ผ ์ƒ์„ฑยถ
 1{% extends "chat/base.html" %}
 2
 3{% block content %}
 4    <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-6">
 5        {% for room in room_list %}
 6            <div class="bg-white rounded-lg shadow-md overflow-hidden hover:shadow-lg transition-shadow duration-300">
 7                <img src="https://placehold.co/600x400/e2e8f0/475569?text=Chat+Room" alt="์ฑ„ํŒ…๋ฐฉ ์ด๋ฏธ์ง€" class="w-full h-48 object-cover">
 8                <div class="p-4">
 9                    <h2 class="text-xl font-semibold text-gray-800 mb-2">{{ room.name }}</h2>
10                    <p class="text-gray-600 text-sm mb-4">์ƒ์„ฑ์ผ: {{ room.created_at|date:"Y-m-d H:i" }}</p>
11                    <a href="{% url 'chat:room_detail' room.pk %}"
12                    class="inline-block w-full text-center bg-indigo-600 text-white py-2 px-4 rounded-md hover:bg-indigo-700 transition-colors duration-300">
13                        ์ž…์žฅํ•˜๊ธฐ
14                    </a>
15                </div>
16            </div>
17        {% empty %}
18            <div class="col-span-full text-center py-8">
19                <p class="text-gray-500 text-lg">์ƒ์„ฑ๋œ ์ฑ„ํŒ…๋ฐฉ์ด ์—†์Šต๋‹ˆ๋‹ค.</p>
20                <a href="{% url 'chat:room_new' %}"
21                class="inline-block mt-4 bg-indigo-600 text-white py-2 px-6 rounded-md hover:bg-indigo-700 transition-colors duration-300">
22                    ์ƒˆ ์ฑ„ํŒ…๋ฐฉ ๋งŒ๋“ค๊ธฐ
23                </a>
24            </div>
25        {% endfor %}
26    </div>
27{% endblock %}
chat/templates/chat/room_detail.html ํŒŒ์ผ ์ƒ์„ฑยถ
 1{% extends "chat/base.html" %}
 2
 3{% block content %}
 4    <div class="flex flex-col h-[calc(100vh-16rem)]">
 5        <div class="bg-white rounded-lg shadow-md p-4 mb-4">
 6            <h1 class="text-2xl font-bold text-gray-800">{{ room.name }}</h1>
 7            <p class="text-sm text-gray-600">์ƒ์„ฑ์ผ: {{ room.created_at|date:"Y-m-d H:i" }}</p>
 8        </div>
 9
10        <div class="flex-1 bg-gray-50 rounded-lg shadow-inner p-4 mb-4 overflow-y-auto">
11            <div class="space-y-4" id="chat-messages">
12                {% for message in message_list %}
13                    <div class="flex {% if message.is_ai %}justify-start{% else %}justify-end{% endif %}">
14                        <div class="{% if message.is_ai %}bg-white{% else %}bg-indigo-600 text-white{% endif %} rounded-lg px-4 py-2 max-w-[80%] shadow">
15                            <div class="text-sm {% if message.is_ai %}text-gray-600{% else %}text-indigo-100{% endif %} mb-1">
16                                {{ message.is_ai|yesno:"AI,You" }}
17                            </div>
18                            <p class="break-words">{{ message.content }}</p>
19                            <div class="text-xs {% if message.is_ai %}text-gray-400{% else %}text-indigo-200{% endif %} text-right mt-1">
20                                {{ message.created_at|date:"H:i" }}
21                            </div>
22                        </div>
23                    </div>
24                {% endfor %}
25            </div>
26        </div>
27
28        <form class="flex gap-2" method="post">
29            {% csrf_token %}
30            <input type="text" name="content" required autocomplete="off"
31                class="flex-1 rounded-lg border-gray-300 shadow-sm focus:border-indigo-300 focus:ring focus:ring-indigo-200 focus:ring-opacity-50"
32                placeholder="๋ฉ”์‹œ์ง€๋ฅผ ์ž…๋ ฅํ•˜์„ธ์š”...">
33            <button type="submit"
34                class="bg-indigo-600 text-white px-6 py-2 rounded-lg hover:bg-indigo-700 transition-colors duration-300">
35                ์ „์†ก
36            </button>
37        </form>
38    </div>
39{% endblock %}
chat/templates/chat/room_form.html ํŒŒ์ผ ์ƒ์„ฑยถ
 1{% extends "chat/base.html" %}
 2
 3{% block content %}
 4<div class="max-w-2xl mx-auto">
 5    <div class="bg-white rounded-lg shadow-md p-6">
 6        <h1 class="text-2xl font-bold text-gray-800 mb-6">์ƒˆ ์ฑ„ํŒ…๋ฐฉ ๋งŒ๋“ค๊ธฐ</h1>
 7
 8        <form method="post" novalidate>
 9            {% csrf_token %}
10
11            <div class="mb-4">
12                <label for="{{ form.name.id_for_label }}" class="block text-sm font-medium text-gray-700 mb-2">์ฑ„ํŒ…๋ฐฉ
13                    ์ด๋ฆ„</label>
14                <input type="text" name="{{ form.name.name }}" id="{{ form.name.id_for_label }}" required
15                    class="w-full rounded-lg border-0 shadow-sm focus:border-indigo-300 focus:ring focus:ring-indigo-200 focus:ring-opacity-50"
16                    {% if form.name.value %}value="{{ form.name.value }}" {% endif %}>
17                {% if form.name.errors %}
18                <p class="mt-2 text-sm text-red-600">{{ form.name.errors.0 }}</p>
19                {% endif %}
20            </div>
21
22            <div class="mb-4">
23                <label for="{{ form.system_prompt.id_for_label }}" class="block text-sm font-medium text-gray-700 mb-2">์‹œ์Šคํ…œ
24                    ํ”„๋กฌํ”„ํŠธ</label>
25                <textarea name="{{ form.system_prompt.name }}" id="{{ form.system_prompt.id_for_label }}"
26                    class="w-full rounded-lg border-0 shadow-sm focus:border-indigo-300 focus:ring focus:ring-indigo-200 focus:ring-opacity-50"
27                    {% if form.system_prompt.value %}value="{{ form.system_prompt.value }}" {% endif %}></textarea>
28                {% if form.system_prompt.errors %}
29                <p class="mt-2 text-sm text-red-600">{{ form.system_prompt.errors.0 }}</p>
30                {% endif %}
31            </div>
32
33            <div class="flex justify-end">
34                <button type="submit"
35                    class="bg-indigo-600 text-white px-6 py-2 rounded-lg hover:bg-indigo-700 transition-colors duration-300">
36                    ์ƒ์„ฑํ•˜๊ธฐ
37                </button>
38            </div>
39        </form>
40    </div>
41</div>
42{% endblock %}

์›น ํŽ˜์ด์ง€ ๋™์ž‘ ํ™•์ธยถ

์•„๋ž˜ ๋ช…๋ น์œผ๋กœ ์žฅ๊ณ  ๊ฐœ๋ฐœ ์›น์„œ๋ฒ„๋ฅผ ๊ตฌ๋™ํ•ฉ๋‹ˆ๋‹ค.

python manage.py runserver 0.0.0.0:8000

ํฌ๋กฌ์ด๋‚˜ ์—ฃ์ง€ ๋ธŒ๋ผ์šฐ์ €๋ฅผ ํ†ตํ•ด http://localhost:8000 ์ฃผ์†Œ๋กœ ์ ‘์†ํ•ด์ฃผ์„ธ์š”.

../../_images/runserver.png

๊ทธ๋Ÿผ mysite/urls.py์— ์ •์˜๋œ RedirectView์— ์˜ํ•ด /chat/ ์ฃผ์†Œ๋กœ ์ž๋™ ์ด๋™ํ•ฉ๋‹ˆ๋‹ค. โ€œ์ƒˆ ์ฑ„ํŒ…๋ฐฉ ๋งŒ๋“ค๊ธฐโ€ ๋ฒ„ํŠผ์„ ํด๋ฆญํ•˜์‹œ๋ฉด /chat/new/ ์ฃผ์†Œ๋กœ ์ด๋™ํ•ฉ๋‹ˆ๋‹ค.

../../_images/room_list1.png

http://localhost:8000/chat/new/ ์ฃผ์†Œ์—์„œ๋Š” ์ƒˆ๋กœ์šด ์ฑ„ํŒ…๋ฐฉ์„ ์ƒ์„ฑํ•  ์ˆ˜ ์žˆ๋Š” ํผ์ด ์ œ๊ณต๋ฉ๋‹ˆ๋‹ค. โ€œ์ฑ„ํŒ…๋ฐฉ ์ด๋ฆ„โ€๊ณผ ์ฑ„ํŒ…๋ฐฉ์—์„œ ์‚ฌ์šฉํ•  โ€œ์‹œ์Šคํ…œ ํ”„๋กฌํ”„ํŠธโ€๋ฅผ ์ž…๋ ฅํ•˜๊ณ  โ€œ์ƒ์„ฑํ•˜๊ธฐโ€ ๋ฒ„ํŠผ์„ ํด๋ฆญํ•˜๋ฉด ์ฑ„ํŒ…๋ฐฉ์ด ์ƒ์„ฑ๋ฉ๋‹ˆ๋‹ค.

../../_images/room_new.png

์ฐธ๊ณ 

chat/templates/chat/room_form.html ํ…œํ”Œ๋ฆฟ์—์„œ๋Š” ํผ ํ•„๋“œ๋งˆ๋‹ค ์ผ์ผ์ด HTML ๋งˆํฌ์—…์„ ์ž‘์„ฑํ–ˆ์ง€๋งŒ, django-crispy-forms ๋“ฑ์˜ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ๋ฅผ ํ†ตํ•ด ํผ ํ•„๋“œ์— ๋Œ€ํ•œ ๋งˆํฌ์—…์„ ์ž๋™์œผ๋กœ ์ƒ์„ฑํ•  ์ˆ˜ ์žˆ๊ณ , HTML ์ฝ”๋“œ๊ฐ€ ์•„๋‹Œ ํŒŒ์ด์ฌ ์ฝ”๋“œ๋กœ ๋ ˆ์ด์•„์›ƒ์„ ๊ตฌ์„ฑํ•˜๊ณ  ์œ„์ ฏ์„ ์ง€์ •ํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.

์ฑ„ํŒ…๋ฐฉ ์ƒ์„ฑ ํ›„์— ์ž๋™์œผ๋กœ ์ฑ„ํŒ…๋ฐฉ ๋ชฉ๋ก ํŽ˜์ด์ง€๋กœ ์ด๋™ํ–ˆ์Šต๋‹ˆ๋‹ค. โ€œ์ž…์žฅํ•˜๊ธฐโ€ ๋ฒ„ํŠผ์„ ํด๋ฆญํ•˜์‹œ๋ฉด ์ฑ„ํŒ…๋ฐฉ ์ฑ„ํŒ… ํŽ˜์ด์ง€๋กœ ์ด๋™ํ•ฉ๋‹ˆ๋‹ค.

../../_images/room_list2.png

http://localhost:8000/chat/1/ ์ฃผ์†Œ์—์„œ๋Š” ์ฑ„ํŒ…๋ฐฉ ์ฑ„ํŒ… ํŽ˜์ด์ง€๊ฐ€ ์ œ๊ณต๋ฉ๋‹ˆ๋‹ค. ์ฑ„ํŒ…๋ฐฉ์— ๋”ฐ๋ผ /chat/1/, /chat/2/, /chat/3/ ๋“ฑ ์ฃผ์†Œ๊ฐ€ ๋‹ฌ๋ผ์ง‘๋‹ˆ๋‹ค. ์•„์ง ์ฑ„ํŒ… ๋ฉ”์‹œ์ง€ ์ „์†ก ๋ฐ ์‘๋‹ต ๊ธฐ๋Šฅ์ด ๊ตฌํ˜„๋˜์ง€ ์•Š์•˜์Šต๋‹ˆ๋‹ค.

../../_images/room_detail.png