Python - Django를 이용한 인스타그램 클론 - 4
이전 단계에서는 글 작성 모달을 띄우는 부분 까지 완료하였다. 이제 작성한 폼이 공유하기 버튼을 누르면 새로운 피드로 추가되게 해야한다.
즉, 우리가 작성한 내용이 장고 내 DB에 들어가서 조회할 수 있도록 해야한다는 것이다.
그말인 즉슨 장고의 MTV패턴으로 설명해보자면 Template에서 받은 파일을 View에서 처리하여 Model 즉 DB 피드테이블의 데이터 타입에 맞게 저장하여야 한다는 말이다.
일반적으로 템플릿(웹)에서 데이터를 받는 방식으로 get방식도 있지만 ajax라는 방식도 있는데, ajax는 Asynchronous JavaScript and XML
의 줄임말으로써, ajax는 HTML의 반복적인 렌더를 방지한다. 즉, 기존 웹 어플리케이션에는 폼을 채우고 이를 제출하게 되면 처음 창을 띄우는 html과 제출 시 폼에 이용되는 html이 중복되는 경우가 많아서 대역폭이 낭비되는 것인데 Ajax는 이와 다르게 어플리케이션에 필요한 데이터만 요청 받을 수 있기 때문에 이러한 단점을 극복할 수 있다.여기서 동기와 비동기의 차이점을 알 수 있는데, 동기 방식에서는 db에서 html파일을 보내고 띄울 수 있을 때 까지 클라이언트는 대기를 해야하지만, ajax와 같이 비동기인 경우에는 응답을 기다리지 않고 사용자의 다른 행동을 허용한다. 우리가 쓰는 인스타그램 피드 생성은 비동기식으로 이루어진다.
우선, 피드를 구성하는 요소로 이미지, 글 내용, 글쓴이 아이디, 글쓴이 프로필 사진 네개가 있다. 피드 테이블은 좋아요 수도 포함하지만 어차피 모든 피드는 처음 생성 이후에는 좋아요 수가 0으로 정해져있으므로 피드생성으로 전달해오는 데이터로 보지 않아도 된다.
이를 구현하기 위해
우선, 공유하기 버튼에 id값을 넣어준다. 필자는 "feed_create_button" 으로 지정해주었다.그리고 이후 Jquery를 작성하는 부분에서
$('#feed_create_button').click(function(){
alert("버튼이 눌렸다");
});
이렇게 해놓고, 웹을 띄우고 공유하기를 누르면,버튼이 눌렸다고 alert가 뜰 것이다.
우리가 해야할 것은 alert가 아니라 서버호출을 해야하는 것이므로, 이를 위해서는 우리가 작성한 피드 데이터를 가져올 수 있도록 이전에 작성한 코드 중
function uploadFiles(e) {
e.stopPropagation();
e.preventDefault();
e.dataTransfer = e.originalEvent.dataTransfer; // 올린 파일을 업로드 하는
var files = e.target.files || e.dataTransfer.files;
console.log("뭔가 이미 파일을 올렸네 " + files[0].name); // 파일을 드래그 해서 여러개 올릴 수도 있기에 files내 리스트처럼 접근해야한다
if (files.length > 1) {
alert('하나만 올려라.');
return;
}
위 업로드 파일 부분에서 var files = ~ 부분이 필요하다. 즉 files라는 변수를 전역변수로 선언해주어야 한다.
그래서 jquery 작성 부분 가장 상단에
let files;
해주고 이전에 있었던 uploadFiles부분의 files는 이미 선언되었으므로 자료형 var을 없애주면 된다.
이렇게 하고 시험삼아 alert와 함께 첫 번째 file의 이름이 출력되게 해보았다
$('#feed_create_button').click(function(){
alert("공유하기를 눌렀다." + files[0].name);
});
그렇게 하고 실행하게 되면
잘 작동하는 것을 알 수 있다. 즉 우리가 올린 데이터를 받아오고 다시 출력 되었다는 것이므로 잘 받아왔다는 것을 알 수 있다.
이렇게 이미지 이름은 잘 불러왔는데, textarea부분은 어떻게 가져올까. 우선 textarea에 id값으로 'input_feed_content'를 주었다.
<textarea id="input_feed_content" style="width: 276px; height: 400px" class="form-control" id="exampleFormControlTextarea1" rows="5"></textarea>
그리고 이후 Jquery 부분에서 해당 id의 value를 가져오는 아래 코드를 삽입해 주었다.
$('#feed_create_button').click(function() {
let file = files[0];
let image = file.image;
let content = $('input_feed_content').val();
let user_id = "Jin.99";
let profile_image = "";
// 우선 user_id와 profile_image는 우리가 html에 데이터로 불러온게 아닌 그냥 삽입해둔 상태라 다른 방식을 거치지 않고 그대로 작성해주었다.
그리고 ajax로 데이터를 저장하는 코드는 아래와 같다.
let fd = new FormData();
fd.append('file', file);
fd.append('image', image);
fd.append('content', content);
fd.append('user_id', user_id);
fd.append('profile_image', profile_image);
// form data에 데이터 추가
$.ajax({
url: "/content/upload",
data: fd,
method: "post",
processData: false,
contentType: false,
success: function(data) {
console.log("성공");
},
error: function(request, status, error){
console.log("에러");
},
complete: function(){
console.log("완료");
}
})
});
ajax는 form-data방식으로 데이터를 가져오는데, 그렇기에 new FormData() 해서 폼 데이터에 가져온 여러 데이터를 append해주었고 아래 $.ajax에서 데이터를 content/upload에 보내는 ajax코드를 썼다. 하지만 우리는 아직 contnet/upload 폴더를 만들지 않지 않았는가, 그럼에도 우선 실행해보았다. 우선 ajax는 성공 시, 에러 시, 완수 시 세 개의 콜백 함수를 둔다. processData나 contentType은 그냥 디폴트로 저렇게 둬도 되는 것 같다.
여기까지 정상적으로 했다면 아마 에러가 떠야 할 것이다.
에러라고 잘 출력한다. 404는 서버에 만들어놓은 경로가 없기에 뜨는 것이다.
이후, content의 views.py에 들어가 클래스를 하나 생성해준다. 우리가 template으로 부터 받은 데이터를 view가 DB에 넣어줘야 하므로 views에서의 작업이 필요한 것이다.
from django.shortcuts import render
from rest_framework.response import Response
from rest_framework.views import APIView
from .models import Feed
#.은 현 폴더에서, models내 Feed 클래스를 사용한다는 것
class Main(APIView):
def get(self, request):
feed_list = Feed.objects.all().order_by('-id') #feed_list에 Feed에 있는 모든 데이터를 가져오겠다는 것
# 위 같이 Feed.objects.all()과 같은 라인이 쿼리셋 역할을 한다. 즉 sql쿼리 문에서 select * from content_feed와 같은 동작을 한다
for feed in feed_list:
print(feed.content)
return render(request, "minstagram/main.html", context=dict(feeds=feed_list)) # 딕셔너리 형태로 feed_list 데이터를 넘겨주기
class UploadFeed(APIView):
def post(self, request):
file = request.data.get('file')
image = request.data.get('image')
content = request.data.get('content')
user_id = request.data.get('user_id')
profile_image = request.data.get('profile_image')
return Response(status=200) #http response -- http:200(=success를 의미)
위 UploadFeed 클래스 처럼 template ajax에서 post로 받아온 각 파일들을 변수로 선언해준다.
그리고 urls.py에 들어가서
from django.contrib import admin
from django.urls import path
from .views import Sub
from content.views import Main, UploadFeed
urlpatterns = [
path('admin/', admin.site.urls),
path('main/', Main.as_view()),
path('content/upload', UploadFeed.as_view())
]
위와 같이 content/upload 경로를 만들어준다. (여기서 UploadFeed 클래스를 import해와야 한다).
그런데 이상하지 않은가, 우리는 이전에 DB에 파일을 올릴 때 파일 그대로 올리면 용량문제와 그에 따른 응답속도 저하 문제 등의 이유로 직접올리지 않고 이미지 파일 같은 경우는 경로를 사용해서 올리곤 했다. 그렇다면 이미지 파일들은 어디로 가야하는 것일까. 대다수 상용화된 서비스에는 이미지서버, 파일서버가 따로 존재하고 거기에 있는 파일의 주소를 DB가 참조를 한다. 장고에서는 이러한 기능으로 media를 제공하고 이 곳에 파일을 저장할 수 있도록 한다.
미디어 파일 세팅은 구글링을 해서 찾아 볼 수 있다. mychew님은 https://wayhome25.github.io/django/2017/05/10/media-file/
장고 미디어 파일 (Media Files) - 사진업로드, 파일서빙 · 초보몽키의 개발공부로그
제출된 파일, 이미지는 settings.MEDIA_ROOT 경로에 파일을 저장하고, DB 필드에는 settings.MEDIA_ROOT 내 저장된 하위 경로를 저장
wayhome25.github.io
위 블로거분의 글을 참고했다.
세팅 방법은, 아래 코드를 settings.py 가장 하단에 붙여넣는다.
# mysite/settings.py
# 각 media 파일에 대한 URL Prefix
MEDIA_URL = '/media/' # 항상 / 로 끝나도록 설정
# MEDIA_URL = 'http://static.myservice.com/media/' 다른 서버로 media 파일 복사시
# 업로드된 파일을 저장할 디렉토리 경로
MEDIA_ROOT = os.path.join(BASE_DIR, 'media') # 프로젝트 경로 + 'media' 이름으로 MEDIA_ROOT라고 생성
os에서 빨간줄은 import해주면 된다.
그리고 프로젝트 폴더에 media폴더를 하나 생성해준다.
그리고 아래 코드를 urls.py에 넣어주자.
urlpatterns += static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
<urls.py>
from django.contrib import admin
from django.urls import path
from django.conf import settings
from django.conf.urls.static import static
from .views import Sub
from content.views import Main, UploadFeed
urlpatterns = [
path('admin/', admin.site.urls),
path('main/', Main.as_view()),
path('content/upload', UploadFeed.as_view())
]
urlpatterns += static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
그리고 views.py에
# 일단 파일 불러와
file = request.FILES['file']
uuid_name = uuid4().hex # 이미지에 고유한 id부여를 위해 랜덤하게 글자를 정함
save_path = os.path.join(MEDIA_ROOT, uuid_name)
with open(save_path, "wb+") as destination:
for chunk in file.chunks():
destination.write(chunk)
내용을 삽입해준다.
Upload Feed 클래스의
file = request.data.get('file')
바로 위에 넣어주면 된다.
그리고, 우리가 피드를 조회했을 때 Feed.objects.all 했던 것 처럼 Feed objects를 생성할 때는 Feed.objects.create를 쓰면 된다. 그래서 Feed를 구성하는 필드, column인 image, content 등 마다 저렇게 변수를 넣어주면 된다.
class UploadFeed(APIView):
def post(self, request):
# 일단 파일 불러와
file = request.FILES['file']
uuid_name = uuid4().hex # 이미지에 고유한 id부여를 위해 랜덤하게 글자를 정함
save_path = os.path.join(MEDIA_ROOT, uuid_name)
with open(save_path, "wb+") as destination: # with open~ 부분이 실제로 파일을 저장하는 부분, save_path에 파일을 열어 조각조각 가져와 쓴다 - 그냥 알고만 있기
for chunk in file.chunks(): # 파일변수명.chunks로 가져올 수 있다잉
destination.write(chunk)
# 이렇게 저장된 경로를 담아주기 - 이미지에
image = uuid_name
content = request.data.get('content')
user_id = request.data.get('user_id')
profile_image = request.data.get('profile_image')
Feed.objects.create(image=image, content=content, user_id=user_id, profile_image=profile_image, like_count=0)
return Response(status=200) #http response -- http:200(=success를 의미)
이렇게 하면 views.py파일은 아래와 같아 질 것이다.
import os
from uuid import uuid4
from django.shortcuts import render
from rest_framework.response import Response
from rest_framework.views import APIView
from Minstagram.settings import MEDIA_ROOT
from .models import Feed
#.은 현 폴더에서, models내 Feed 클래스를 사용한다는 것
class Main(APIView):
def get(self, request):
feed_list = Feed.objects.all().order_by('-id') #feed_list에 Feed에 있는 모든 데이터를 가져오겠다는 것
# 위 같이 Feed.objects.all()과 같은 라인이 쿼리셋 역할을 한다. 즉 sql쿼리 문에서 select * from content_feed와 같은 동작을 한다
for feed in feed_list:
print(feed.content)
return render(request, "minstagram/main.html", context=dict(feeds=feed_list)) # 딕셔너리 형태로 feed_list 데이터를 넘겨주기
class UploadFeed(APIView):
def post(self, request):
# 일단 파일 불러와
file = request.FILES['file']
uuid_name = uuid4().hex # 이미지에 고유한 id부여를 위해 랜덤하게 글자를 정함
save_path = os.path.join(MEDIA_ROOT, uuid_name)
with open(save_path, "wb+") as destination: # with open~ 부분이 실제로 파일을 저장하는 부분, save_path에 파일을 열어 조각조각 가져와 쓴다 - 그냥 알고만 있기
for chunk in file.chunks(): # 파일변수명.chunks로 가져올 수 있다잉
destination.write(chunk)
# 이렇게 저장된 경로를 담아주기 - 이미지에
image = uuid_name
content = request.data.get('content')
user_id = request.data.get('user_id')
profile_image = request.data.get('profile_image')
Feed.objects.create(image=image, content=content, user_id=user_id, profile_image=profile_image, like_count=0)
return Response(status=200) #http response -- http:200(=success를 의미)
이렇게 해서 실행을 해보면, 글 작성이 된다. 하지만 탑재한 이미지가 뜨지 않는데, 이미지 파일 명만 잡고 앞부분에 파일 서버 경로를 제대로 잡지 않아서이다.
그래서 main.html에서 head부분에
{% load static %}
을 넣어주고, 사진이 보여지는 곳인 {{ feed.image }} 앞에
<div><img style="width:100%; height: 100%" src="{% get_media_prefix %}{{ feed.image }}">
{% get_media_prefix %}를 추가해주면 서버 경로를 앞에 추가해 잘 뜨게 된다.
그럼 우선 이전에 만들어 놓은 피드 데이터를 모두 삭제하고, 추가해보도록 하겠다.
데이터 삭제는 그냥 데이터를 싸잡아 delete해주고 업로드하면 된다.
이렇게 하고, 공유하기
잘 올라갔다. 보우가 귀엽다. 뭐 댓글이나 이런건 html 소스코드에 추가된 부분이라 해당 부분도 수정해주면 되는 부분이다.
그리고 공유하기 버튼이 누르면 모달이 사라지지 않고 가만히 있어서 새로고침을 해야하는데, 자동으로 닫기도록 설정해주자.
ajax로 정보가 전달되는게 완료되면 location을 main으로, 즉 페이지를 새로고침 해주게 코드를 추가해주었다.
요렇게 내용을 추가하고 공유하기 누르면!
새로고침 되며 모달창이 사라지고, 피드가 조회되었다.