virtualenv 설치 및 가상환경 세팅
pip install virtualenv
#버전 확인을 통해서 설치 확인
virtualenv --version
#
virtualenv flask_web & cd flask_web
가상환경 활성화
Scripts\activate
(flask_web) C:\apps\flask_web>
pip install flask
필요한 라이브러 설치후 관리 (pip 명령어)
pip freeze > requirements.txt
requirements.tx 에 기록된 라이브러리 설치
pip install -r requirements.txt
서버를 띄우기 위해 다음과 같이 app.py를 생성후 다음과 같은 코드를 작성한다.
from flask import Flask
app = Flask(__name__)
app.debug = True
if __name__ == '__main__':
app.run()
클라이언트가 http://localhost:5000/data 으로 GET방식으로 요청이 들어왔을때 "Hello World !" 문구를 리턴해 본다.
그러기 위해서 다음과 같은 코드를 추가한다.
from flask import Flask
.....
@app.route('/data', methods=['GET'])
def index():
return "Hello World"
.....
render_template 메소드를 이용해서 index.html을 화면에 랜더링 해본다.
templates/index.html 을 생성후 다음과 같이 코드를 추가한다.
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
{{data}}
<h1>Hello World!!</h1>
</body>
</html>
app.py의 def index(): 을 수정한다.
@app.route('/data', methods=['GET'])
def index():
# return "Hello World"
return render_template("index.html", data="KIM")
render_template 은 첫번째 인자로 html파일 경로 , 두번째 인자로 전달할 데이터를
jinja2 엔진 문법
{{ ... }} : 변수나 표현식의 결과를 출력하는 구분자(delimeter)
{% ... %} : if문이나 for문 같은 제어문을 할당하는 구분자(delimeter)
{# ... #} : 주석
{%- ... %}, {%+ ... %}, {% ... -%} ... : 공백 제거 혹은 유지
{% raw %} ... {% endraw %} : 이스케이프
예)
{% for <개별요소> in <리스트> %}
<실행코드>
{% endfor %}
클라이언트가 http://localhost:5000 으로 GET방식으로 요청이 들어왔을때 index.html 랜더링 되도록 다음과 같은 코드를 추가한다.
from flask import Flask , render_template
.....
@app.route('/', methods=['GET'])
def index():
# return "Hello World"
return render_template("index.html", data="KIM")
.....
index.html 파일을 만든후 다음과 같이 코딩을 한다.
그런데 페이지마다 공통으로 적용되는 부분은 따로 만들어서 extends 한다.
공통으로 적용되는 파일을 layouts.html 을 생성후 다음과 같이 코딩한다.
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.5.0/css/bootstrap.min.css" integrity="sha384-9aIt2nRpC12Uk9gS9baDl411NQApFmC26EwAOH8WgZl5MYYxFfc+NcPb1dKGj7Sk" crossorigin="anonymous">
<title>Flask_Web</title>
</head>
<body>
{% include "includes/_navbar.html" %}
<div class="container">
<!-- Content here -->
{% block body %}
{% endblock %}
</div>
<script src="https://code.jquery.com/jquery-3.5.1.slim.min.js" integrity="sha384-DfXdz2htPH0lsSSs5nCTpuj/zy4C+OGpamoFVy38MVBnE+IbbVYUew+OrCXaRkfj" crossorigin="anonymous"></script>
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/umd/popper.min.js" integrity="sha384-Q6E9RHvbIyZFJoft+2mJbHaEWldlvI9IOYy5n3zV9zzTtmI3UksdQRVvoxMfooAo" crossorigin="anonymous"></script>
<script src="https://stackpath.bootstrapcdn.com/bootstrap/4.5.0/js/bootstrap.min.js" integrity="sha384-OgVRvuATP1z7JjHLkuOU7Xw704+h835Lr+6QL9UvYjZE3Ipu6Tp75j7Bh/kR0JKI" crossorigin="anonymous"></script>
</body>
</html>
{% block body %}
...... 부분에 들어갈 콘텐츠를 만든다.
{% endblock %}
navigation bar를 다시 include 한다.
includes/_navbar.html 을 생성
<nav class="navbar navbar-expand-lg navbar-light bg-warning">
<a class="navbar-brand" href="/">Flask_Web</a>
<button class="navbar-toggler" type="button" data-toggle="collapse" data-target="#navbarNav" aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
<span class="navbar-toggler-icon"></span>
</button>
<div class="collapse navbar-collapse" id="navbarNav">
<ul class="navbar-nav">
<li class="nav-item active">
<a class="nav-link" href="/">Home
<span class="sr-only">(current)</span>
</a>
</li>
<li class="nav-item">
<a class="nav-link" href="/about">About</a>
</li>
<li class="nav-item">
<a class="nav-link" href="/articles">Articles</a>
</li>
</ul>
</div>
</nav>
index.html 다음과 같이 코드를 생성한다.
{% extends "layouts.html" %}
{% block body %}
<h1 class="display-4">HOME PAGE</h1>
<p class="lead">This Application is build using by python and Flask Framework</p>
<hr class="my-4">
<a class="btn btn-warning btn-lg" href="#" role="button">Learn more</a>
{% endblock %}
{% block body %}
..... 생성
{% endblock %}
다음과 같은 페이자 완성 되었다.
http://localhost:5000/about 페이지 구성을 위해서 app.py 에 다음과 같은 코드를 추가한다.
...
@app.route('/about')
def about():
return render_template("about.html", hello = "Gary Kim")
...
about.html 파일 생성은 코드 생성
{% extends "layouts.html" %}
{% block body %}
<h1 class="display-4">ABOUT PAGE</h1>
<p class="lead">This Application is build using by python and Flask Framework</p>
<hr class="my-4">
<a class="btn btn-warning btn-lg" href="#" role="button">Learn more</a>
{% endblock %}
다음과 같은 페이지 완성
http://localhost:5000/articles 페이지 구성을 위해서 app.py 에 다음과 같은 코드를 추가한다.
...
@app.route('/articles')
def articles():
return render_template("articles.html", hello = "Gary Kim")
...
about.html 파일 생성은 코드 생성
{% extends "layouts.html" %}
{% block body %}
<h1 class="display-4">ARTICLE PAGE</h1>
<p class="lead">This Application is build using by python and Flask Framework</p>
<hr class="my-4">
<a class="btn btn-warning btn-lg" href="#" role="button">Learn more</a>
{% endblock %}
다음과 같은 페이지 완성
data.py파일에 임의의 MOCKDATA를 만들어 articles에 데이터로 전달하기 위해
data.py
def Articles():
articles = [ { 'id': 1, 'title':'Article one', 'body':'Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.', 'author':'vasanth', 'create_date':'04-09-2018', },
{ 'id': 2, 'title':'Article two', 'body':'Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.', 'author':'vasanth nagarajan', 'create_date':'05-09-2018', },
{ 'id': 3, 'title':'Article three', 'body':'Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.', 'author':'nagarajan vasanth', 'create_date':'04-09-2018', } ]
return articles
app.py 파일에 다음과 같은 코드 추가
...
from data import Articles
...
@app.route('/artcles')
def articles():
articles = Articles()
# print(articles[0]['title'])
return render_template("articles.html", articles = articles)
articles 변수로 데이터 전달
artcles.html을 jinja2문법을 사용하여 for문으로 작성
{% extends "layouts.html" %}
{% block body %}
<h1 class="display-4">ARTICLES PAGE</h1>
<p class="lead">This Application is build using by python and Flask Framework</p>
<hr class="my-4">
<table class="table" style="width:100% ">
<thead class="thead-dark">
<tr>
<th scope="col">ID</th>
<th scope="col">TITLE</th>
<th scope="col">description</th>
<th scope="col">AUTHOR</th>
<th scope="col" >DATE</th>
<th scope="col" >EDIT</th>
</tr>
</thead>
<tbody>
{% for article in articles %}
<tr>
<th scope="col" >{{article['id']}}</th>
<td>{{article['title']}}</td>
<td>{{article['body']}}</td>
<td>{{article['author']}}</td>
<td>{{article['create_date']}}</td>
</tr>
{% endfor %}
</tbody>
</table>
<a class="btn btn-warning btn-lg" href="#" role="button">Learn more</a>
{% endblock %}
pip install PyMySQL
설치후
import pymysql 실행
- dbmake.py
import pymysql
db = pymysql.connect(
host = 'localhost',
port = 3306,
user = 'root',
passwd = '1234',
db = 'busan'
)
# cursor = db.cursor()
# db.commit()
# cursor.execute('SELECT * FROM users;')
# users = cursor.fetchall()
# print(users)
- dbmake.py
sql = '''
CREATE TABLE `topic` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`title` varchar(100) NOT NULL,
`body` text NOT NULL,
`author` varchar(30) NOT NULL,
`create_date` TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (id)
) ENGINE=innoDB DEFAULT CHARSET=utf8;
'''
# cursor = db.cursor()
# cursor.execute(sql)
# db.commit()
# users = cursor.fetchall()
# print(users)
- dbmake.py
sql_1 = "INSERT INTO `busan`.`topic` (`title`, `body`, `author`) VALUES ('부산', '부산와서 갈매기를 못봤네ㅠㅠ', '정연우');"
cursor = db.cursor()
cursor.execute(sql_1)
db.commit()
users = cursor.fetchall()
print(users)
- dbmake.py
sql_2 = "INSERT INTO `busan`.`users` (`name`, `email`, `username`, `password`) VALUES ('Jung', '[email protected]', 'Jung', '12345');"
cursor = db.cursor()
cursor.execute(sql_2)
db.commit()
cursor.execute('SELECT * FROM busan.users;')
users = cursor.fetchall()
print(users)
※ fetchall()은 튜플안에 튜플형태 ex) : ( () , () , ... )
- dbmake.py
sql_3 = "INSERT INTO `topic` (`title`, `body`, `author`) VALUES (%s, %s, %s);"
cursor = db.cursor()
title = input('제목을 적으세요 : ')
body = input('내용을 적으세요 : ')
author = input('저자를 적으세요 : ')
input_data = [title, body, author ]
cursor.execute(sql_3,input_data)
db.commit()
cursor.execute('SELECT * FROM topic;')
users = cursor.fetchall()
print(cursor.rowcount, users)
- app.py
import pymysql
from flask import Flask , render_template, request, redirect
from data import Articles
app = Flask(__name__)
app.debug = True # 내폴더구조가 모두보임 기본값은 false false로 해놓으면 안보임
db = pymysql.connect(
host = 'localhost'
post = 3306
user = 'root'
passwd = '1234'
db = 'busan'
)
'''
.
.
.
'''
# 변경
@app.route('/articles')
def articles():
# articles = Articles()
# print(articles[0]['title'])
cursor = db.cursor()
sql = 'SELECT * FROM topic;'
cursor.execute(sql)
topics = cursor.fetchall() # 튜플 가로 두개라 for문 필수
print(topics)
return render_template("articles.html", articles=topics)
- articles.html
<tbody>
{% for article in articles %}
<tr>
<th scope="row">{{ article[0] }}</th>
<td><a href="/article/{{article[0]}}">{{ article[1] }}</a></td>
<td>{{ article[2] }}</td>
<td>{{ article[3] }}</td>
<td>{{ article[4] }}</td>
<td><button type="button" class="btn btn-danger">Edit</button></td>
</tr>
{% endfor %}
</tbody>
- result !
- app.py
@app.route('/article/<int:id>') #params
def article(id):
# articles = Articles()
# article = articles[id - 1]
# print(articles[id - 1])
cursor = db.cursor()
sql = 'SELECT * FROM topic where id = {};'.format(id)
cursor.execute(sql)
topic = cursor.fetchone()
print(topic)
return render_template("article.html", article = topic)
※ fetchone()은 튜플 한개 형태 ex) : ( )
- article.html
{% extends "layouts.html" %}
{% block body %}
<div class="card">
<div class="card-header">
{{ article[3] }}
</div>
<div class="card-body">
<h5 class="card-title">{{ article[1] }}</h5>
<p class="card-text">{{ article[2] }}</p>
<a href="/articles" class="btn btn-primary">돌아가기</a>
</div>
</div>
<a class="btn btn-warning btn-lg" href="/" role="button">Home</a>
{% endblock %}
- result!
app.py
@app.route('/add_articles', methods=['GET', 'POST'])
def add_articles():
cursor = db.cursor()
if request.method == "POST":
author = request.form['author']
title = request.form['title']
desc = request.form['desc']
sql = "INSERT INTO `topic` (`title`, `body`, `author`) VALUES (%s, %s, %s);"
input_data = [title, desc, author]
cursor.execute(sql, input_data)
db.commit()
print(cursor.rowcount)
return redirect('/articles')
else:
return render_template('add_articles.html')
- add_articles.html
{% extends "layouts.html" %}
{% block body %}
<form action="/add_articles" method="post">
<input type="text" name="author" placeholder="작성자" required>
<input type="text" name="title" placeholder="제목" required>
<input type="text" name="desc" placeholder="내용" required>
<input type="submit" value="작성">
</form>
{% endblock %}
- result!
-
articles.html
맨 마지막 코드 변경 필요 주소를 add_articles로 바꿔줌으로써 글쓰기 누르면
add_articles로 가짐
btn-warning btn-lg" href="/add_articles" role="button">글쓰기</a>
- result!
- app.py
@app.route('/delete/<int:id>', methods=["POST"])
def delete(id):
cursor = db.cursor()
# sql = "DELETE FROM topic WHERE id = %s;" #format쓰면 밑에 2줄 id 필요없음
# id = [id]
# cursor.execute(sql , id)
sql = 'DELETE FROM topic WHERE id = {};'.format(id)
cursor.execute(sql)
db.commit()
return redirect('/articles')
- articles.html
<td>
<button type="button" class="btn btn-danger">Edit</button>
<form action="/delete/{{articles[0]}}" style="margin-top: 10px;" method="POST" onsubmit="return confirm('정말 삭제 하시겠습니까?')">
<button type="submit" class="btn btn-danger">Delete</button>
</form>
</td>
-
articles.html
css 변경해줌
{% extends 'layouts.html' %}
{% block body %}
<div class="main-container">
<!-- horizontal Basic Forms Start -->
<div class="pd-20 card-box mb-30">
<div class="clearfix">
<div class="pull-left">
<h4 class="text-blue h4">Writing Lists Board</h4>
<p class="mb-30">Input your Words !!</p>
</div>
</div>
<form action="/add_articles" method="POST" onsubmit="return confirm('이데로 작성하시겠습니까?')">
<div class="form-group">
<label>TITLE</label>
<input class="form-control" type="text" name='title' placeholder="input your title" required>
</div>
<div class="form-group">
<label>DESCRIPTON</label>
<textarea class="form-control" name="desc" placeholder="input your description" required></textarea>
</div>
<div class="form-group">
<label>AUHOR</label>
<input type="text" name='author' class="form-control" placeholder="input your name" required>
</div>
<div class="form-group">
<input class="btn btn-warning btn-lg btn-block" type="submit" value="작성완료" >
</div>
<div class="form-group">
<a class="btn btn-primary btn-lg btn-block" href="/articles"> 돌아가기</a>
</div>
</form>
</div>
<!-- horizontal Basic Forms End -->
</div>
{% endblock %}
-
result!
이런식으로 변경됨
- app.py
@app.route('/<int:id>/edit', methods=["GET", "POST"])
def edit(id):
if request.method == "POST":
return "Success"
else:
return render_template('edit_article.html')
- articles.html
<a href="/{{article[0]}}/edit">
<button type="button" class="btn btn-danger">Edit</button>
</a>
edit_article.html
{% extends "layouts.html" %}
{% block body %}
<h1 class="display-4">ABOUT PAGE</h1>
<p class="lead">This Application is build using by python and Flask Framework</p>
<hr class="my-4">
<form
action="/edit"
method="post"
onsubmit="return confirm('정말 수정하시겠습니까?')"
<input type="text" name = "title" required>
<input type="text" name = "desc" required>
<input type="text" name = "author" disabled>
<input type="submit" value="수정하기">
>
</form>
<a class="btn btn-warning btn-lg" href="/articles" role="button">돌아가기</a>
{% endblock %}
- result!
- app.py
@app.route('/<int:id>/edit', methods=["GET", "POST"])
def edit(id):
cursor = db.cursor()
if request.method == "POST":
return "Success"
else:
sql = 'SELECT * FROM topic WHERE id = {}'.format(id)
cursor.execute(sql)
topic = cursor.fetchone()
print(topic, '\n')
print(topic[1])
return render_template('edit_article.html', article = topic)
- edit_article.html
<input type="text" name = "title" value="{{article[1]}}" required>
<input type="text" name = "desc" value="{{article[2]}}" required>
<input type="text" name = "author" value="{{article[3]}}" disabled>
<input type="submit" value="수정하기">
- result!
- app.py
@app.route('/<int:id>/edit', methods=["GET", "POST"])
def edit(id):
cursor = db.cursor()
if request.method == "POST":
title = request.form['title']
desc = request.form['desc']
sql = "UPDATE topic SET title = %s, body = %s WHERE id = {};".format(id)
input_data = [title, desc]
cursor.execute(sql, input_data)
db.commit()
print(request.form['title'])
return redirect('/articles')
else:
sql = 'SELECT * FROM topic WHERE id = {}'.format(id)
cursor.execute(sql)
topic = cursor.fetchone()
# print(topic, '\n')
# print(topic[1])
return render_template('edit_article.html', article = topic)
- edit_article.html
action="/{{article[0]}}/edit"
- result!
edit_article.html
<div class="jumbotron text-left">
<div class="main-container">
<!-- horizontal Basic Forms Start -->
<div class="pd-20 card-box mb-30">
<div class="clearfix">
<div class="pull-left">
<h4 class="text-blue h4">Edit Your List</h4>
</div>
</div>
<form action="/{{article[0]}}/edit" method="POST" onsubmit="return confirm('이제로 편집하시겠습니까?')">
<div class="form-group">
<label>TITLE</label>
<input class="form-control" type="text" name='title' value="{{article[1]}}" required>
</div>
<div class="form-group">
<label>DESCRIPTON</label>
<textarea class="form-control" name="desc" rows="10" required>{{ article[2]}}</textarea>
</div>
<div class="form-group">
<label>AUHOR</label>
<input type="text" name='author' class="form-control" value="{{article[3]}}" required>
</div>
<div class="form-group">
<input class="btn btn-warning btn-lg btn-block" type="submit" value="편집완료" >
</div>
<div class="form-group">
<a class="btn btn-primary btn-lg btn-block" href="/articles"> 돌아가기</a>
</div>
</form>
</div>
<!-- horizontal Basic Forms End -->
</div>
<hr class="my-4">
</div>
-
result
hash코드 뜨는거 확인 가능
- app.py
userpw = request.form['userpw']
-> userpw = sha256_crypt.encrypt(request.form['userpw'])
- app.py
@app.route('/login', methods = ["GET", "POST"])
def login():
cursor = db.cursor()
if request.method == "POST":
usersname = request.form['username']
userpw_1 = request.form['userpw']
sql = 'SELECT password FROM users WHERE email = %s;'
input_data = [usersname]
cursor.execute(sql, input_data)
userpw = cursor.fetchone()
# print(userpw[0])#cursor.fetchone()[0]
print(userpw[0])
if sha256_crypt.verify(userpw_1, userpw[0]):
return "Success"
else:
return userpw[0]
- register.html
<input class="form-control" type="password" name='userpw' required>
- app.py
@app.route('/register', methods=["POST","GET"])
def register():
cursor = db.cursor()
if request.method == "POST":
name = request.form['name']
email = request.form['email']
username = request.form['username']
# userpw = (request.form['userpw'])
userpw = sha256_crypt.encrypt(request.form['userpw'])
sql = "INSERT INTO users (name, email, username, password) VALUES (%s, %s, %s, %s);"
input_data = [name, email, username, userpw]
cursor.execute(sql, input_data)
db.commit()
return redirect('/')
else:
return render_template("register.html")
@app.route('/login', methods = ["GET", "POST"])
def login():
cursor = db.cursor()
if request.method == "POST":
email = request.form['email']
userpw_1 = request.form['userpw']
# print(request.form['username'])
# print(userpw_1)
sql = 'SELECT * FROM users WHERE email = %s;'
input_data = [email]
cursor.execute(sql, input_data)
user = cursor.fetchone()
# print(userpw[0])#cursor.fetchone()[0]
if user == None:
print(user)
return redirect('/register')
else:
if sha256_crypt.verify(userpw_1, user[4]):
# 앞뒤 맞는지 안 맞는지 검증
return redirect('/articles')
else:
return user[4]
else:
return render_template('login.html')
- login.html
{% extends "layouts.html"%}
<!-- 진자 엔진의 import방법 = extends -->
{% block body %}
<form action="/login" method="POST">
<div class="form-group">
<label>ID</label>
<input type="email" name="email" class="form-control" placeholder="ID">
</div>
<div class="form-group">
<label>PW</label>
<input type="password" name="userpw" class="form-control" placeholder="Password">
</div>
<input class="btn btn-dark btn-dark" type="submit" value="Login">
</form>
<h1 class="display-4">MAIN HOME PAGE</h1>
<p class="lead">This Application is build using by python and Flask Framework</p>
<!-- <a class="btn btn-dark btn-dark" href="/assignment" role="button">Assignment</a> -->
<hr class="my-4">
<a class="btn btn-warning btn-dark" href="/register" role="button">회원가입</a>
{% endblock %}