29 Tue

TIL

[ํ”„๋กœ๊ทธ๋ž˜๋จธ์Šค AI ์Šค์ฟจ 1๊ธฐ] 4์ฃผ์ฐจ DAY 2

AWS๋ฅผ ํ™œ์šฉํ•œ ์ธ๊ณต์ง€๋Šฅ ๋ชจ๋ธ ๋ฐฐํฌ III

API to serve ML model

Client๊ฐ€ HTTP ํ”„๋กœํ† ์ฝœ์„ ํ†ตํ•ด ์„œ๋ฒ„์— ์š”์ฒญํ–ˆ์„ ๊ฒฝ ํ•™์Šต ๋˜๋Š” ์˜ˆ์ธก์„ ํ•  ์ˆ˜ ์žˆ๋Š” API๋ฅผ ๋งŒ๋“œ๋Š” ๊ฒƒ์ด ๋ชฉํ‘œ. ์ด ๋ชจ๋ธ์€ ํ•™์Šต์„ ํ†ตํ•ด ๋‚ด๋ถ€ EBS์— ์ €์žฅ๋˜๊ฑฐ๋‚˜ ์™ธ๋ถ€์˜ ์‚ฌ์ „์— ํ•™์Šตํ•œ ๋ชจ๋ธ์„ ๋ฒ„ํ‚ท์œผ๋กœ ์ €์žฅํ•˜๊ฑฐ๋‚˜ ๊นƒํ—ˆ๋ธŒ ์ €์žฅ์†Œ์—์„œ ๊ฐ€์ ธ์˜ฌ ์ˆ˜๋„ ์žˆ๋‹ค.

Interface

์‚ฌ์šฉ์ž๊ฐ€ ๊ธฐ๊ณ„์™€ ์†Œํ”„ํŠธ์›จ์–ด๋ฅผ ์ œ์–ดํ•˜๊ธฐ ์œ„ํ•ด ์ƒํ˜ธ ํ•ฉ์˜๋œ ๋ฉ”๋‰ด์–ผ

ex) ๋ฆฌ๋ชจ

API

Application Programming Interface. ์‚ฌ๋žŒ์ด ๊ธฐ๊ณ„๋ฅผ ์ œ์–ดํ•˜๋Š” ๊ฒƒ์ฒ˜๋Ÿผ. ๊ธฐ๊ณ„์™€ ๊ธฐ๊ณ„, ์†Œํ”„ํŠธ์›จ์–ด์™€ ์†Œํ”„ํŠธ์›จ์–ด ๊ฐ„์˜ ์ปค๋ฎค๋‹ˆ์ผ€์ด์…˜์„ ์œ„ํ•œ ์ธํ„ฐํŽ˜์ด์Šค

ex) ํœด๋Œ€ํฐ์— ๋„์ฐฉํ•œ ์นดํ†ก ์•Œ๋ฆผ์„ ์›Œ์น˜์— ์ถœ๋ ฅ

RESTful API for ML/DL model inference

REST์•„ํ‚คํ…์ฒ˜๋ฅผ ๋”ฐ๋ฅด๋Š” API์ด๋ฉฐ HTTP URI๋ฅผ ํ†ตํ•ด ์ž์›์„ ๋ช…์‹œํ•˜๊ณ  HTTP Method๋ฅผ ํ†ตํ•ด ํ•„์š”ํ•œ ์—ฐ์‚ฐ์„ ์š”์ฒญํ•˜๊ณ  ๋ฐ˜ํ™˜ํ•˜๋Š” API๋ฅผ ์ง€์นญ

Practical process of machine learning

๊ฒ€์ฆ์„ ํ†ตํ•ด ๋ฐฐํฌ๊ฐ€ ๊ฐ€๋Šฅํ•˜๋‹ค๊ณ  ์ƒ๊ฐํ•˜๋ฉด ๋ฐฐํฌ. ์—ฌ๊ธฐ์„œ ์šฐ๋ฆฌ๊ฐ€ ํ•  ๋ถ€๋ถ„์€ Trained Model File์„ ๋ถˆ๋Ÿฌ์™€์„œ ์‹คํ–‰ํ•˜๋Š” ๊ฒƒ.

Model Serving

๋ชจ๋ธ์„ ์ œ์ž‘, ํ›ˆ๋ จ, ํ‰๊ฐ€ํ•œ๋‹ค. ์ดํ›„ ์™„์„ฑ๋œ ๋ชจ๋ธ์„ ์ €์žฅํ•˜๊ณ  ์ดํ›„์— ๋‹ค์‹œ ๋ถˆ๋Ÿฌ์˜จ๋‹ค. ์ด ๊ณผ์ •์—์„œ ์—ฐ์†์„ฑ์ด ์กด์žฌํ•ด์•ผ ํ•œ๋‹ค.

Serialization & De-serialization

S : ML/DL model์„ disk์— writeํ•˜์—ฌ loadable ํ•  ์ˆ˜ ์žˆ๋Š” ํ˜•ํƒœ๋กœ ๋ณ€ํ™˜.

D : Python๋“ฑ์˜ ํ™˜๊ฒฝ์—์„œ model์„ ๋ถˆ๋Ÿฌ์™€ ์ถ”๋ก /ํ•™์Šต.

Handler

์ฝ”๋“œ์˜ ๊ด€๋ฆฌ์ธก๋ฉด์—์„œ ํ•ธ๋“ค๋Ÿฌ๊ฐ€ ํ•„์š”

Frameworks for serving

๋”ฅ๋Ÿฌ๋‹ ํ™˜๊ฒฝ์—์„œ๋Š” GPU๋‚˜ ๋ถ„์‚ฐ ์ฒ˜๋ฆฌ ํ™˜๊ฒฝ์„ ๊ฐ€์ง€๊ณ  ์•ˆ์ •์ ์œผ๋กœ ์ œ๊ณตํ•˜๋Š” ๊ฒƒ์ด ์ค‘์š”. Tensorflow serving์ด๋‚˜ TorchServe, TensorRT ๊ฐ™์€ ํ”„๋ ˆ์ž„์›Œํฌ๋ฅผ ์‚ฌ์šฉํ•˜๋Š” ๊ฒƒ์ด ์ผ๋ฐ˜์ ์ด๋‹ค.

์‹ค์Šต : Serialization & De-serialization

[ํ”„๋กœ๊ทธ๋ž˜๋จธ์Šค AI ์Šค์ฟจ 1๊ธฐ] 4์ฃผ์ฐจ DAY 4

django IV - Model

view ์—์„œ๋Š” ๋กœ์ง์ฒ˜๋ฆฌ๊ฐ€ ์žˆ์—ˆ๊ณ  ์ด ์ฒ˜๋ฆฌ์˜ ๋ฐ˜ํ™˜๊ฐ’์ด ์ฒ˜๋ฆฌ์— ๋Œ€ํ•œ ๋ฐ˜ํ™˜๊ฐ’์ด๋‹ค. ์ด ๋•Œ http response๋ฅผ ์‚ฌ์šฉํ•  ์ˆ˜๋„ ์žˆ์ง€๋งŒ render๋ฅผ ์‚ฌ์šฉํ•  ์ˆ˜๋„ ์žˆ๋‹ค. ์ด ๋•Œ ํ…œํ”Œ๋ฆฟ์„ ์‚ฌ์šฉํ•˜๋ฉฐ ํ…œํ”Œ๋ฆฟ์€ html, css, js๊ฐ€ ์žˆ๋‹ค.

model์€ aws๋ฅผ ๊ด€๋ฆฌํ•˜๋Š” ๊ณณ์ธ๋ฐ, ๋ชจ๋ธ์„ ํ™œ์šฉํ•ด์„œ ํ…œํ”Œ๋ฆฟ ๋˜๋Š” ๋ทฐ์— ์œ ์˜๋ฏธํ•œ ์ •๋ณด๋ฅผ ์ œ๊ณตํ•œ๋‹ค. ๋˜, ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค๋„ ๊ด€๋ฆฌํ•˜๊ฒŒ ๋œ๋‹ค.

๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค๋Š” ๋ฐ์ดํ„ฐ๋ฅผ ๊ตฌ์กฐํ™” ํ•˜์—ฌ ์ €์žฅํ•œ ์‹œ์Šคํ…œ์ด๋‹ค. ๋‹จ์ˆœํžˆ ์ •๋ณด๋ฅผ ์ €์žฅํ•œ ์ฐฝ๊ณ ๊ฐ€ ์•„๋‹Œ, ์ •๋ ฌ๊ธฐ๋Šฅ์„ ์ œ๊ณตํ•˜์—ฌ ์‰ฝ๊ฒŒ ๊ด€๋ฆฌํ•  ์ˆ˜ ์žˆ๊ฒŒ ํ•œ๋‹ค. Relational DB๋ฅผ ๋งŽ์ด ์‚ฌ์šฉํ•˜๋ฉฐ ์ด๋Š” column๊ณผ row๋ฅผ ์ด์šฉํ•ด ๋ฐ์ดํ„ฐ๋ฅผ ์ •์˜ํ•œ๋‹ค. ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค์— ์ ‘๊ทผํ•  ์ˆ˜ ์žˆ๋Š” API๊ฐ€ SQL์ด๋‹ค. ๊ทธ๋Ÿฌ๋‚˜ ์žฅ๊ณ ๋Š” SQL๋ฅผ ์ด์šฉํ•˜์ง€ ์•Š๊ณ ๋„ ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค์— ์ ‘๊ทผํ•  ์ˆ˜ ์žˆ๋Š”๋ฐ ์ด๊ฒƒ์ด ORM์ด๋‹ค. ์žฅ๊ณ ์—๋Š” ๊ฐ์ฒด ๋‹จ์œ„๋กœ ๋ฐ์ดํ„ฐ๋ฅผ ๋‹ค๋ฃจ๋Š” ORM์ด ๋‚ด์žฅ๋˜์–ด์žˆ๋‹ค.

models.py

from django.db import models

# Create your models here.
class Coffee(models.Model):
    name = models.CharField(default="", null=False, max_length=30)    # default : ๊ธฐ๋ณธ์ ์œผ๋กœ ํ–‰์— ๋‹ด๊ฒจ์žˆ๋Š” ๊ฐ’
    price = models.IntegerField(default=0)       # null : ๋น„์–ด์žˆ์–ด๋„ ๋˜๋Š”์ง€์˜ ์—ฌ๋ถ€ (default : False)
    is_ice = models.BooleanField(default=False)     # max_length : ์ตœ๋Œ€ ๊ธธ์ด
    '''
    ๋ฌธ์ž์—ด : CharField
    ์ˆซ์ž : IntegerField, SmallIntegerField
    ๋…ผ๋ฆฌํ˜• : BooleanField
    ์‹œ๊ฐ„/๋‚ ์งœ : DateTimeField
    '''

admin.py

from django.contrib import admin
from .models import Coffee

# Register your models here.
# ์–ด๋–ค ๋ชจ๋ธ์„ admin๊ณผ ์—ฐ๋™์„ ํ•˜๋ฉด superuse์—์„œ ์ด ํŽ˜์ด์ง€๋ฅผ ๊ด€๋ฆฌํ•  ์ˆ˜ ์žˆ์Œ

admin.site.register(Coffee)

๊ด€๋ฆฌ์ž ํŽ˜์ด์ง€๋กœ ์ ‘์†ํ•˜๋ฉด Coffees๊ฐ€ ์ถ”๊ฐ€๋œ ๋ชจ์Šต. ๊ธฐ๋ณธ์ ์œผ๋กœ Groups์™€ Users๋ฅผ ๊ด€๋ฆฌํ•œ๋‹ค. ์ด ๋•Œ coffees๋ฅผ ๋ˆ„๋ฅด๋ฉด ์˜ค๋ฅ˜๊ฐ€ ๋ฐœ์ƒํ•˜๋Š”๋ฐ ์ด๋Š” ์•„์ง ์žฅ๊ณ  ํ™˜๊ฒฝ์—์„œ coffees๊ฐ€ ์ƒ์„ฑ๋จ์„ ์ธ์‹ํ•˜์ง€ ๋ชปํ–ˆ๊ธฐ ๋•Œ๋ฌธ.

python manage.py makemigrations homepage

python manage.py migrate

๋ฅผ ์‹คํ–‰ํ•˜๊ณ  ๋‹ค์‹œ coffees๋ฅผ ๋ˆ„๋ฅด๋ฉด ๋‹ค์Œ๊ณผ ๊ฐ™์€ ํ™”๋ฉด์ด ์„ฑ๊ณต์ ์œผ๋กœ ์ถœ๋ ฅ๋œ๋‹ค.

name์€ charfield์ด๊ธฐ ๋•Œ๋ฌธ์— ๊ธด ํ…์ŠคํŠธ ๋ฐ•์Šค, price๋Š” intergerfield up-down ๊ธฐ๋Šฅ์ด ๋‚ด์žฅ๋œ ํ…์ŠคํŠธ ๋ฐ•์Šค, is ice๋Š” booleanfield์ด๊ธฐ ๋•Œ๋ฌธ์— ์ฒดํฌ ๋ฐ•์Šค๊ฐ€ ์ƒ์„ฑ๋œ ๊ฒƒ์„ ์•Œ ์ˆ˜ ์žˆ๋‹ค.

์ด ๋•Œ ์œ„์™€ ๊ฐ™์ด Coffee ํด๋ž˜์Šค๋กœ ์ƒ์„ฑ๋œ ๋‘ ๊ฐœ์˜ ์ธ์Šคํ„ด์Šค๊ฐ€ ์กด์žฌํ•˜์ง€๋งŒ ๊ฐœ๋ณ„์ ์ธ ์ด๋ฆ„์„ ๊ฐ€์งˆ ํ•„์š”๊ฐ€ ์žˆ๋‹ค. ์ด๋ฅผ ์œ„ํ•ด ๋‹ค์Œ์˜ ์ฝ”๋“œ๋ฅผ Coffee ํด๋ž˜์Šค์— ์ถ”๊ฐ€ํ•œ๋‹ค.

    def __str__(self):
        return self.name

์œ„์™€ ๊ฐ™์ด ๋ณ€๊ฒฝ๋œ ๋ชจ์Šต.

django V - Model on Template

๋ชจ๋ธ์˜ ๋‚ด์šฉ์„ ํ…œํ”Œ๋ฆฟ์œผ๋กœ ์ „๋‹ฌํ•˜๊ณ ์ž ํ•˜๋Š”๋ฐ ์ด ๋•Œ ํ•ญ์ƒ view๋ฅผ ๊ฑฐ์ณ์•ผ ํ•œ๋‹ค.

coffee.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Coffee List</title>
</head>
<body>
    <h1>My Coffee List</h1>
    {% for coffee in coffee_list %}
        <p>{{ coffee.name }} , {{ coffee.price }}</p>
    {% endfor %}
</body>
</html>

urls.py

from django.contrib import admin
from django.urls import path
from homepage.views import index, coffee_view

urlpatterns = [
    path('coffee/', coffee_view), # 127.0.0.1/coffee/
    path('admin/', admin.site.urls), # 127.0.0.1/admin/
    path('', index) # 127.0.0.1/
]

์ƒˆ๋กœ์šด ํŒŒ์ผ์ด ์ƒ๊ธธ ๋•Œ ๋งˆ๋‹ค ํ•ญ์ƒ ์žฅ๊ณ ์˜ ๊ตฌ์กฐ์— ๋งž๊ฒŒ ์ถ”๊ฐ€์‹œ์ผœ์ฃผ๊ธฐ

views.py

def coffee_view(request):
    coffee_all = Coffee.objects.all()   # ์ปคํ”ผ ํด๋ž˜์Šค์˜ ๋ชจ๋“  ํ–‰์„ ๊ฐ€์ ธ์˜ค๋ผ๋Š” ๋œป.
    return render(request, 'coffee.html', {"coffee_list" : coffee_all})

์ด ๊ณผ์ •์€ ๋‹ค์Œ๊ณผ ๊ฐ™์ด ์ง„ํ–‰๋œ๋‹ค.

model.py : ์ž˜ ๋งŒ๋“ค์–ด์ง„ ๋ชจ๋ธ ํŒŒ์ผ

view.py : ๋ชจ๋ธ์„ import๋ฅผ ํ•˜๊ณ  ๋ชจ๋ธ์˜ ๋ฐ์ดํ„ฐ๋ฅผ ๊ฐ€์ง€๊ณ  ์˜จ๋‹ค. ์ด ๋•Œ, objects.all() ์„ ํ†ตํ•ด ๊ฐ€์ง€๊ณ  ์˜ด. ์ด ์™ธ์—๋„ get()์ด๋‚˜ filter()๋“ฑ์„ ํ†ตํ•ด ํŠน์ • ์กฐ๊ฑด์„ ๋งŒ์กฑํ•˜๋Š” ๋ฐ์ดํ„ฐ๋ฅผ ๊ฐ€์ง€๊ณ  ์˜ฌ ์ˆ˜ ์žˆ์Œ. ์ดํ›„์— render๋ฅผ ์ด์šฉํ•˜์—ฌ dictionary ํ˜•ํƒœ๋กœ template์— ๋„˜๊ฒจ์ค€๋‹ค.

coffee.html : ์ด ์ •๋ณด๋ฅผ ๋„˜๊ฒจ ๋ฐ›์€ ํ…œํ”Œ๋ฆฟ์€ ํ…œํ”Œ๋ฆฟ ๋ณ€์ˆ˜์™€ ํƒœ๊ทธ๋ฅผ ํ†ตํ•ด ์ •๋ณด๋ฅผ ๋™์ ์œผ๋กœ ์ฒ˜๋ฆฌํ•œ๋‹ค.

django VI - Updating Model on Template with Form

Form์„ ์ด์šฉํ•ด์„œ ํ…œํ”Œ๋ฆฟ์˜ ๋ชจ๋ธ์„ ์ˆ˜์ •ํ•˜๊ธฐ

forms.py

from django import forms
from .models import Coffee  # Model ํ˜ธ์ถœ

class CoffeeForm(forms.ModelForm):  # ModelForm์„ ์ƒ์†๋ฐ›๋Š” CoffeeForm ์ƒ์„ฑ
    class Meta:
        model = Coffee
        fields = ('name', 'price', 'is_ice')

views.py

def coffee_view(request):
    coffee_all = Coffee.objects.all()   # ์ปคํ”ผ ํด๋ž˜์Šค์˜ ๋ชจ๋“  ํ–‰์„ ๊ฐ€์ ธ์˜ค๋ผ๋Š” ๋œป.
    form = CoffeeForm()
    return render(request, 'coffee.html', {"coffee_list" : coffee_all, "coffee_form" : form})

๋งˆ์ฐฌ๊ฐ€์ง€๋กœ view ๊ฑฐ์ณ ํ…œํ”Œ๋ฆฟ์œผ๋กœ ์ •๋ณด๊ฐ€ ์ „์†ก๋˜์–ด์•ผ ํ•œ๋‹ค.

coffee.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Coffee List</title>
</head>
<body>
    <h1>My Coffee List</h1>
    {% for coffee in coffee_list %}
        <p>{{ coffee.name }} , {{ coffee.price }}</p>
    {% endfor %}

    <form method="Post">{% csrf_token %}
        {{ coffee_form.as_p }}
        <button type="submit">Save</button>
    </form>
</body>
</html>

form์„ ๊ตฌ์„ฑํ•  ๋•Œ๋Š” CSRF ํ† ํฐ์„ ๊ผญ ๋„ฃ์–ด์ค˜์•ผ ํ•œ๋‹ค. HTML5 ๋ณด์•ˆ์ •์ฑ…์ด๋‹ค. ๊ทธ๋ ‡์ง€ ์•Š์œผ๋ฉด ์•„๋ž˜์™€ ๊ฐ™์€ ์—๋Ÿฌ ํŽ˜์ด์ง€๊ฐ€ ์ถœ๋ ฅ๋œ๋‹ค.

์ด ๋•Œ save๋ฒ„ํŠผ์„ ๋ˆŒ๋Ÿฌ POST์š”์ฒญ์„ ํ•ด๋„ ์•„๋ฌด๋Ÿฐ ๋ณ€ํ™”๊ฐ€ ์—†๋‹ค. ์ด์— ๋Œ€ํ•œ ๋กœ์ง ์ฒ˜๋ฆฌ๋ฅผ ๊ตฌํ˜„ํ•ด์•ผ ํ•จ.

views.py

def coffee_view(request):
    coffee_all = Coffee.objects.all()   # ์ปคํ”ผ ํด๋ž˜์Šค์˜ ๋ชจ๋“  ํ–‰์„ ๊ฐ€์ ธ์˜ค๋ผ๋Š” ๋œป.
    if request.method == "POST":
        form = CoffeeForm(request.POST)    # ์–‘์‹์ด ์ฑ„์›Œ์ง„ Form ์ƒ์„ฑ
        if form.is_valid():     # is_valid๋Š” form ๋‚ด๋ถ€ ๊ฐ’๋“ค์ด ์œ ํšจํ•œ์ง€ ํ™•์ธํ•œ๋‹ค
            form.save()     # Form ๋‚ด์šฉ์„ ๋ชจ๋ธ์— ์ €์žฅ
    form = CoffeeForm()
    return render(request, 'coffee.html', {"coffee_list" : coffee_all, "coffee_form" : form})

Last updated

Was this helpful?