5 Fri

[HEAD FIRST PYTHON] 5๊ฐ• ์›น์•ฑ ๋งŒ๋“ค๊ธฐ

Flask ์„ค์น˜

  • pip install flask

  • ๊ฐ€์ƒํ™˜๊ฒฝ์—์„œ ์„ค์น˜

  • ์ด ๋•Œ flask๊ฐ€ ์˜์กด์„ฑ์„ ๊ฐ–๋Š” ๋„ค ๊ฐœ์˜ ๋ชจ๋“ˆ์ด ํ•จ๊ป˜ ์„ค์น˜๋œ๋‹ค.

    • Successfully installed Jinja2-2.11.3 MarkupSafe-1.1.1 Werkzeug-1.0.1 click-7.1.2 flask-1.1.2 itsdangerous-1.1.0

Flask

  • ๋งˆ์ดํฌ๋กœ ์›น ํ”„๋ ˆ์ž„์›Œํฌ

    • ํƒœ์Šคํฌ์— ํ•„์š”ํ•œ ์ตœ์†Œํ•œ์˜ ๊ธฐ์ˆ ์ง‘ํ•ฉ์„ ์ œ๊ณต

    • ๊ฒฝ์Ÿ์ž์ธ ์žฅ๊ณ ๋Š” ๋ชจ๋“  ๊ธฐ๋Šฅ์„ ํฌํ•จํ•œ๋‹ค

from flask import Flask
app = Flask(__name__)

@app.route('/')
def hello() -> str:
    return "Hello world from Flask!"

app.run()
  • 1 : f๊ฐ€ ์†Œ๋ฌธ์ž์ธ flask๋Š” ๋ชจ๋“ˆ๋ช…์ด๋ฉฐ F๊ฐ€ ๋Œ€๋ฌธ์ž์ธ Flask๋Š” ํด๋ž˜์Šค๋ช…์ด๋‹ค.

    • import flask.Flask๋„ ๊ฐ€๋Šฅํ•˜์ง€๋งŒ ๊ฐ€๋…์„ฑ์„ ์œ„ํ•ด from - import ๋ฅผ ์‚ฌ์šฉํ•œ๋‹ค.

  • 2 : Flask ํด๋ž˜์Šค์˜ ์ธ์Šคํ„ดํŠธ๋ฅผ ๋งŒ๋“ค๊ณ  ์ด๋ฅผ app์ด๋ผ๊ณ  ์ •์˜ํ–ˆ๋‹ค. Flask ํด๋ž˜์Šค๋Š” ๊ฐ์ฒด๋ฅผ ๋งŒ๋“ค ๋•Œ ํ˜„์žฌ ๋ฒ„์ „์˜ __name__์„ ์ธ์ž๋กœ ๋ฐ›์•„์•ผ ํ•œ๋‹ค.

    • __name__์€ ํ˜„์žฌ ํ™œ์„ฑ ๋ชจ๋“ˆ์˜ ์ด๋ฆ„

    • __name__๊ณผ ๊ฐ™์€ ๋ณ€์ˆ˜๋Š” ๋‘ ๊ฐœ์˜ ์–ธ๋”์Šค์ฝ”์–ด๊ฐ€ ์–‘์˜†์— ์กด์žฌํ•˜์—ฌ ๋”๋ธ” ์–ธ๋”์Šค์ฝ”์–ด๋ผ๊ณ  ํ•œ๋‹ค. ์ด๋Š” ์ค„์—ฌ์„œ ๋˜๋” ๋„ค์ž„, dunder name ์ด๋ผ๊ณ ๋„ ํ•œ๋‹ค.

  • 4 : ์žฅ์‹์ž๋ฅผ ์‚ฌ์šฉํ–ˆ๋‹ค.

    • ์žฅ์‹์ž, decorator๋Š” ๊ธฐ์กด ์ฝ”๋“œ์— ์ƒˆ๋กœ์šด ๋™์ž‘์„ ์ถ”๊ฐ€ํ•œ๋‹ค. ์ด ๋•Œ ํ•จ์ˆ˜์˜ ์ฝ”๋“œ๋ฅผ ๋ฐ”๊พธ์ง€ ์•Š๊ณ ๋„ ๋™์ž‘์„ ์กฐ์ ˆํ•  ์ˆ˜ ์žˆ๋‹ค. ํ•จ์ˆ˜๋ฟ ์•„๋‹ˆ๋ผ ํด๋ž˜์Šค์—๋„ ์žฅ์‹์ž๋ฅผ ์ ์šฉํ•  ์ˆ˜ ์žˆ๋‹ค. ์ฃผ๋กœ ํ•จ์ˆ˜์— ์ ์šฉํ•˜๊ธฐ ๋•Œ๋ฌธ์— ํ•จ์ˆ˜ ์žฅ์‹์ž๋ผ๊ณ  ๋งŽ์ด ํ•œ๋‹ค.

    • @ ๊ธฐํ˜ธ๋กœ ์‹œ์ž‘ํ•œ๋‹ค.

    • ์ด ๋•Œ URL '/' ๋Š” ๋‹ค์Œ ํ–‰์— ์žˆ๋Š” ํ•จ์ˆ˜์™€ ์ด์–ด์ง„๋‹ค. URL ์š”์ฒญ์ด ๋“ค์–ด์™”์„ ๋•Œ ์–ด๋–ค ํ•จ์ˆ˜๋ฅผ ํ˜ธ์ถœํ•  ๊ฒƒ์ธ์ง€ ์กฐ์ •ํ•œ๋‹ค. ์ด ๋•Œ ์žฅ์‹์ž๋Š” ํ•จ์ˆ˜์—์„œ ๊ฒฐ๊ณผ๋ฅผ ๋ฐ˜ํ™˜ํ•  ๋•Œ ๊นŒ์ง€ ๋Œ€๊ธฐํ•˜๋‹ค๊ฐ€ ๊ฒฐ๊ณผ๊ฐ€ ๋ฐ˜ํ™˜๋˜๋ฉด ๋Œ€๊ธฐ ์ค‘์ธ ์›น ๋ธŒ๋ผ์šฐ์ €๋กœ ๊ฒฐ๊ณผ๋ฅผ ๋ฐ˜ํ™˜ํ•œ๋‹ค.

    • '/' URL๋กœ ๋“ค์–ด์˜ค๋Š” ๋ชจ๋“  ์„œ๋ฒ„ ์š”์ฒญ์—๋Š” ํ•จ์ˆ˜๋ฅผ ๋ฐ˜ํ™˜ํ•˜๋ฉฐ ๊ทธ ์™ธ์— URL๋กœ ๋“ค์–ด์˜ค๋Š” ์š”์ฒญ์—๋Š” 404 Resource not found ์—๋Ÿฌ ๋ฉ”์‹œ์ง€๋ฅผ ๋ณด๋‚ธ๋‹ค.

  • 8 : ์›น์•ฑ ์‹คํ–‰ ์š”์ฒญ

  • ํ”„๋กœ๊ทธ๋žจ์ด ์‹คํ–‰๋˜๋ฉด ํ”Œ๋ผ์Šคํฌ ์›น ์„œ๋ฒ„๊ฐ€ ์‹คํ–‰๋˜๋ฉฐ ํ”Œ๋ผ์Šคํฌ์˜ ํ…Œ์ŠคํŠธ ์›น ์ฃผ์†Œ(127.0.0.1)์™€ ํ”„๋กœํ† ์ฝœ ํฌํŠธ ๋ฒˆํ˜ธ(5000)๋กœ ์„œ๋น„์Šค ์›น ์š”์ฒญ์„ ๊ธฐ๋‹ค๋ฆฐ๋‹ค๋Š” ๋ฉ”์‹œ์ง€๊ฐ€ ๋‚˜ํƒ€๋‚œ๋‹ค.

์›น์— ๊ธฐ๋Šฅ ์ถ”๊ฐ€ํ•˜๊ธฐ

  • 4์žฅ์—์„œ ๋งŒ๋“ค์—ˆ๋˜ vsearch ๋ชจ๋“ˆ์„ ์ด์šฉํ•ด ๊ธฐ๋Šฅ์„ ์ถ”๊ฐ€ํ•œ๋‹ค.

from flask import Flask
from vsearch import search4letters

app = Flask(__name__)

@app.route('/')
def hello() -> str:
    return "Hello world from Flask!"

@app.route('/search4')
def do_search() -> str:
    return str(search4letters('lif, the universe, and everything', 'eiru'))

app.run()
  • ์ด ๋•Œ app.run(debug=True) ๋กœ ์„ค์ •ํ•˜๊ฒŒ ๋˜๋ฉด, ์ฝ”๋“œ๋ฅผ ๊ณ ์น  ๋•Œ๋งˆ๋‹ค ์žฌ์‹คํ–‰ ํ•˜์ง€ ์•Š์•„๋„ ์›น ์„œ๋ฒ„๊ฐ€ ์•Œ์•„์„œ ๋ฆฌ๋กœ๋“œ ๋œ๋‹ค.

  • 127.0.0.1 ์€ localhost๋ผ ์•Œ๋ ค์ง„ ์ธํ„ฐ๋„ท์˜ ๋ฃจํ”„๋ฐฑ ์ฃผ์†Œ๋ฅผ ์˜๋ฏธํ•œ๋‹ค. ์ด ์ฃผ์†Œ๋Š” IP์™€ ๊ด€๊ณ„์—†์ด ๋‚ด ์ปดํ“จํ„ฐ๋ฅผ ์ง€๋ชฉํ•  ์ˆ˜ ์žˆ๋Š” ๋ฐฉ๋ฒ•์ด๋‹ค.

  • 5000 ์€ ์›น ์„œ๋ฒ„๊ฐ€ ์‹คํ–‰ ์ค‘์ธ ํ”„๋กœํ† ์ฝœ ํฌํŠธ ๋ฒˆํ˜ธ๋ฅผ ์˜๋ฏธํ•œ๋‹ค. ๋ณดํ†ต ์›น์„œ๋ฒ„๋Š” 80์ด๋ผ๋Š” ํฌํŠธ ๋ฒˆํ˜ธ์—์„œ ์‹คํ–‰๋˜๋ฉฐ, ์›น์•ฑ์„ ๊ฐœ๋ฐœํ•˜๋Š” ๋™์•ˆ์—๋Š” 80๋ฒˆ์„ ์“ฐ์ง€ ์•Š๋Š” ๊ฒƒ์ด ๊ด€๋ก€์ด๋‹ค. 8080 ๋ฒˆ ํฌํŠธ๊ฐ€ ๊ฐœ๋ฐœ ์‹œ ์ž์ฃผ ์‚ฌ์šฉํ•˜๋Š” ํฌํŠธ ์ค‘ ํ•˜๋‚˜์ด๋‹ค. ํ”Œ๋ผ์Šคํฌ๋Š” ํ…Œ์ŠคํŠธ ํ”„๋กœํ† ์ฝœ ํฌํŠธ๋กœ 5000 ๋ฒˆ์„ ์‚ฌ์šฉํ•œ๋‹ค.

ํ…œํ”Œ๋ฆฟ ์—”์ง„

  • HTML์„ ์ฝ”๋“œ์— ์ž‘์„ฑํ•˜๋Š” ๊ฒƒ์€ ์ž‘๋™ํ•˜๋Š” ๋ฐ ๋ฌธ์ œ๊ฐ€ ์—†์ง€๋งŒ ํ™•์žฅ์„ฑ์€ ๋–จ์–ด์ง„๋‹ค. ์›น์•ฑ์ด ๋” ์ปค์งˆ์ˆ˜๋ก ๊ด€๋ฆฌํ•˜๊ธฐ๊ฐ€ ์–ด๋ ค์›Œ์ง€๋ฉฐ ์ด๋ฅผ ์›น ๋””์ž์ด๋„ˆ์—๊ฒŒ ๋„˜๊ฒจ์„œ ๊ฐœ์„ ํ•˜๊ธฐ๋„ ์–ด๋ ต๋‹ค

    • ๋”ฐ๋ผ์„œ ํ…œํ”Œ๋ฆฟ ์—”์ง„์„ ์‚ฌ์šฉํ•œ๋‹ค

  • ํ”„๋กœ๊ทธ๋ž˜๋จธ๊ฐ€ ์›น ํŽ˜์ด์ง€ ๊ฐ™์€ ํ…์ŠคํŠธ ๊ธฐ๋ฐ˜ ๋ฐ์ดํ„ฐ ๊ฒฐ๊ณผ๋ฌผ์— ๊ฐ์ฒด ์ง€ํ–ฅ ๊ฐœ๋…์„ ์ ์šฉํ•  ์ˆ˜ ์žˆ๋„๋ก ํ•œ๋‹ค.

  • ๋‹ค์Œ๊ณผ ๊ฐ™์€ ์ตœ์ƒ์œ„ ํ…œํ”Œ๋ฆฟ์„ ์ž‘์„ฑํ•œ๋‹ค. ์ดํ›„ ์—ฌ๋Ÿฌ ํ•˜์œ„ ํ…œํ”Œ๋ฆฟ์ด ์ถ”๊ฐ€๋œ๋‹ค.

<!doctype html>
<html>
    <head>
        <title>{{ the_title }}</title>
        <link rel="stylesheet" href="static/hf.css" />
    </head>
    <body>
        {% block body %}

        {% endblock %}
    </body>
</html>
  • 5 : css๋Š” html์˜ ์Šคํƒ€์ผ์„ ๊ฒฐ์ •ํ•˜๋Š” ํ…์ŠคํŠธ๋กœ ์—ฌ๊ธฐ์„œ๋Š” hf.css๋ฅผ ์‚ฌ์šฉํ•  ๊ฒƒ์œผ๋กœ ๋ช…์‹œํ•˜๊ณ  ์žˆ๋‹ค.

  • 8, 10 : block body ๊ณผ endblock ๋ฅผ ์‚ฌ์šฉํ•˜๋ฏ€๋กœ ํ…œํ”Œ๋ฆฟ์„ ํ™•์žฅํ•  ๊ฒƒ์„ ๋ช…์‹œํ•˜๊ณ  ์žˆ๋‹ค.

  • ๋‹ค์Œ์€ ๊ธฐ๋ณธ ๋ฒ ์ด์Šค ํ…œํ”Œ๋ฆฟ bast.html ์˜ block body์™€ endblock ์‚ฌ์ด์— ๋“ค์–ด๊ฐˆ ์ˆ˜ ์žˆ๋Š” entry.html ๊ณผ results.html ์ด๋‹ค.

{% extends 'base.html' %}

{% block body %}

<h2>{{ the_title }}</h2>

<p>You submitted the following data:</p>
<table>
<tr><td>Phrase:</td><td>{{ the_phrase }}</td></tr>
<tr><td>Letters:</td><td>{{ the_letters }}</td></tr>
</table>

<p>When "{{the_phrase }}" is search for "{{ the_letters }}", the following 
results are returned:</p>
<h3>{{ the_results }}</h3>

{% endblock %}
{% extends 'base.html' %}

{% block body %}

<h2>{{ the_title }}</h2>

<form method='POST' action='/search4'>
<table>
<p>Use this form to submit a search request:</p>
<tr><td>Phrase:</td><td><input name='phrase' type='TEXT' width='60'></td></tr>
<tr><td>Letters:</td><td><input name='letters' type='TEXT' value='aeiou'></td></tr>
</table>
<p>When you're ready, click this button:</p>
<p><input value='Do it!' type='SUBMIT'></p>
</form>

{% endblock %}

HTTP ์ƒํƒœ ์ฝ”๋“œ

  • GET ๋ฉ”์„œ๋“œ

    • ๋ธŒ๋ผ์šฐ์ €๊ฐ€ ์›น ์„œ๋ฒ„๋กœ ๋ฆฌ์†Œ์Šค๋ฅผ ์š”์ฒญํ•˜๋Š” ๋ฉ”์„œ๋“œ

  • POST ๋ฉ”์„œ๋“œ

    • ๋ธŒ๋ผ์šฐ์ €๊ฐ€ HTTP๋ฅผ ํ†ตํ•ด ์„œ๋ฒ„๋กœ ๋ฐ์ดํ„ฐ๋ฅผ ๋ณด๋‚ด๋Š” ๋ฉ”์„œ๋“œ

Render_Template

  • ๋‹ค๋ฅธ ํ…œํ”Œ๋ฆฟ์œผ๋กœ ๋™์  ๋ณ€์ˆ˜๋ฅผ ์ „๋‹ฌํ•  ๋–„ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋‹ค

  • ๋งˆ์ง€๋ง‰์— , ๋ฅผ ๋ช…์‹œํ•ด์ค˜์•ผํ•œ๋‹ค.

return render_template('results.html', the_phrase = phrase, the_letters =letters, the_title=title, the_results=results,)

Redirect

  • redirect๋ฅผ ์ด์šฉํ•ด์„œ URL('/') ์„ URL('/entry')๋กœ redirectํ•œ ๋ชจ์Šต

@app.route('/')
def hello() -> '302':
    return redirect('/entry')
  • ์ด๋Š” ์•„๋ž˜์™€ ๊ฐ™์ด ๋ฐ”๊ฟ”์„œ ์“ธ ์ˆ˜ ์žˆ๋‹ค.

@app.route('/')
@app.route('/search4', methods=['POST'])

์ตœ์ข… ์ฝ”๋“œ

  • vsearch4web.py

from flask import Flask, render_template, request, redirect
from vsearch import search4letters

app = Flask(__name__)

@app.route('/')
@app.route('/search4', methods=['POST'])
def do_search() -> str:
    phrase = request.form['phrase']
    letters = request.form['letters']
    title = 'Here are your results'
    results = str(search4letters(phrase, letters))
    return render_template('results.html', the_phrase = phrase, the_letters =letters, the_title=title, the_results=results,)

@app.route('/entry')
def entry_page() -> 'html':
    return render_template('entry.html', the_title='Welcome to search4letters on the web!')
app.run(debug=True)

app.run()

  • ๋ฐฐํฌ์‹œ์—๋Š” app.run() ์„ ์‹คํ–‰ํ•  ๊ฒฝ์šฐ ์›น์•ฑ ์‹คํ–‰์ด ๊ฑฐ์ ˆ๋œ๋‹ค. ์ด๋Š” ํŒŒ์ด์ฌ์• ๋‹ˆ์›จ์–ด์—์„œ ์ง€์›ํ•˜์ง€ ์•Š๊ธฐ ๋•Œ๋ฌธ.

  • ๊ทธ๋Ÿฌ๋‚˜ ๋””๋ฒ„๊น… ๋ฐ ๊ฐœ๋ฐœ์‹œ์—๋Š” app.run() ์„ ์‚ฌ์šฉํ•˜๊ฒŒ ๋œ๋‹ค. ๋”ฐ๋ผ์„œ ๋‹ค์Œ๊ณผ ๊ฐ™์ด ์ฝ”๋“œ๋ฅผ ์ž‘์„ฑํ•˜๋ฉด ๊ฐœ๋ฐœ์‹œ์—๋งŒ ์‹คํ–‰ํ•˜๊ฒŒ ๋œ๋‹ค.

if __name__ = '__main__':
    app.run(debug=True)

Last updated

Was this helpful?