🔍Storage에 비디오 영상 넣고 가져오기
Firebase에 Realtime base를 이용하여 데이터 넣는 방법은 예전 포스팅에서 한 적이 있다.
(만일 Firebase에 대해 모른다면 아래 포스팅부터 보는 것을 추천한다.)
Realtime base 같은 경우 단순 text 에 적합하지, 영상이나 사진같이 큰 용량의 데이터에는 적합하지 않다.
영상이나 사진을 보관하기위해서는 Storage가 적합하고, 해당 저장 링크를 통해 html에 나타내는게 효율적이다.
따라서 이번 포스팅에는 영상을 저장하고, html에 나타내는 부분까지 할 예정이다.
📚 Storage 시작
Realtime base 블로그 포스팅을 봤다는 전제로 한다.
지난 포스팅에서 만든 프로젝트에서 "Storage" 부분에 들어간다.
Storage에서 "시작하기"를 누르고, 기본 디폴트 값으로 진행하면 된다.
읽고 쓰기를 자유롭게 하기위해 , "규칙"에 들어가서 read와 write를 허용 해준다 (9번 줄 코드처럼 하면 됨)
그 후 "프로젝트 설정" -> "서비스 계정"에 들어간 후,
python의 "새 비공개 키 생성"으로 key를 다운받아준다.
해당키는 servicekey로 json 형태로 저장되어 비디오 read / wirte에 필수이다.
해당 json 파일은 내 프로젝트 폴더에 잘 저장한다.
(지난 포스팅 realtimebase에서 만든 auth 폴더에 같이 보관하는게 좋음)
📚 Video Storage Read / Write
📑 Video Storage class 준비
firebase_recode.py 파일을 새호 만들어 class를 만들어 줄 예정이다. (파일명은 자유롭게 해도 된다.)
import firebase_admin
from firebase_admin import credentials, storage
import pyrebase
import json
from datetime import datetime, timedelta
class Storage :
def __init__(self):
if not firebase_admin._apps:
cred = credentials.Certificate("./auth/serviceAccountKey.json")
firebase_admin.initialize_app(cred, {
'storageBucket': 'emergencyresponse-b8c54.appspot.com',
'databaseURL': 'https://emergencyresponse-b8c54-default-rtdb.firebaseio.com/' # Firebase Realtime Database URL
})
with open("./auth/firebaseAuth.json") as f:
config = json.load(f)
self.firebase = pyrebase.initialize_app(config)
self.storage = self.firebase.storage()
"Firebase Admin SDK"는 서버 사이드에서 Firebase 서비스를 사용하기 위해 Admin SDK를 초기화하고,
다양한 Firebase 기능에 접근 가능
"Pyrebase"는 클라이언트 사이드에서 REST API를 사용하여 사용자 인증 및 데이터베이스 작업에 사용
해당 코드에서 각각의 initialize_app은 다른 목적으로 사용된다.
Firebase Admin SDK 같은 경우 애플리케이션 당 한 번만 초기화 가능하기 때문에 if 문으로 제어했다.
("./auth/serviceAccountKey.json" 이 부분 같은 경우, 아까 "새 비공개 키 생성" 부분에서 저장한 key 경로)
("./auth/firebaseAuth.json" 이 부분 같은 경우, 이전 포스팅에서 저장한 key 경로)
'storageBucket' 부분과 'databaseURL' 부분에서는 Storage에 해당 URL만 복사 붙여 넣으면 되고,
나머지 뒷 부분은 그대로 쓰면 된다.
여기까지 했으면, 준비가 다 된 상태이다.
📑 Video Storage 에 넣기
def video_save(self) :
self.storage.child("Video/recode/1.mp4").put("1.mp4","jerry")
return 0
child("저장하고 싶은 경로")
put("저장하려는 파일 경로" , "UID") 형식으로 넣어주면 된다.
Storage 클래스에서 video_save함수를 실행 시키고,
저장된 것을 확인해보면 Video/recode에 원하는 영상이 잘 저장 된 것을 확인 할 수 있다.
만일 storage에 영상이 잘 재생이 안되면, 바로 위의 포스팅을 참고하자.
대부분 코덱 변환의 문제이다.
📑 Storage에서 Video 꺼내오기
def video_getUrl(self):
bucket = storage.bucket()
blobs = bucket.list_blobs(prefix="Video/recode")
urls = []
for blob in blobs:
url = blob.generate_signed_url(timedelta(seconds=300)) # URL 생성
urls.append(url)
return urls
storage에 저장된 video를 가져오기위해 url을 통해 가져오는게 효과적이다.
bucket() 함수와 blobs 함수를 통해서 원하는 링크를 가져올 수 있다.
가져온 링크들은 urls 리스트에 넣어주고 urls 리스트를 반환 해준다.
지금 현재 Storage에 저장되어있는 파일은 3개이다. (2개 추가함)
Storage 클래스에서 video_getUrl함수를 실행 시켜보면,
3개 비디오에 대한 url이 뜨는 것을 확인 할 수 있다.
📚 Storage Video HTML에 올리기
📑 Flask 서버 사이드
영상을 저장하고 꺼내는 것은 할 줄 알게되었다.
이제 Storage에 저장된 영상을 html에 게시하는 것을 해보자.
app = Flask( __name__ )
DB = DBModule()
STORAGE = Storage()
@app.route("/")
def index():
return render_template("index.html")
#-------------------firebase storage -------------------
@app.route('/video-urls', methods=['GET'])
def video_urls():
urls = STORAGE.video_getUrl()
return jsonify(urls)
flask 서버 기반으로 할 예정이다. (flask 기초 설명 X / flask 같은 경우 templates 구조 필요)
라우팅 할 video_urls 을 GET 형식으로 만들어 주었다.
해당 url은 video_getUrl에서 전달받은 urls 리스트들을 json형태로 반환해주는게 전부이다.
사실 서버 사이드 쪽은 이게 전부이다 .
📑 Frontend 사이드
<h1>Video Gallery</h1>
<div id="video-gallery">
<!-- 비디오 썸네일이 여기에 동적으로 추가. -->
</div>
<div class="video-player" id="video-player">
<video id="video" controls>
<source id="video-source" src="" type="video/mp4">
Your browser does not support the video tag.
</video>
</div>
div "id = video-gallery" 태그에서 Strorage의 영상들을 동적으로 담을 예정이다.
div "class = video-play" 태그는 선택한 비디오를 재생할 비디오 플레이어이다.
source 태그에 비디오 url을 설정하고 controls 속성으로 기본 컨트롤을 추가한다.
<style>
.video-thumbnail {
width: 200px;
height: 150px;
display: inline-block;
margin: 10px;
cursor: pointer;
background-color: #ccc;
display: flex;
align-items: center;
justify-content: center;
font-weight: bold;
}
.video-player {
display: none;
width: 640px;
height: 360px;
margin-top: 20px;
}
</style>
이것은 css적인 요소인데, video-thumbnail과 video-player를 어떤식으로 꾸밀지를 나타냈다.
(video-thumbnail는 Storage와의 통신을 통해서 만들어지는 요소의 class 이다. (밑에서 나올 예정) )
<script>
// 서버에서 비디오 URL 받아오기
fetch('/video-urls')
.then(response => response.json())
.then(videoUrls => {
console.log('Fetched video URLs:', videoUrls); // 디버깅용 콘솔 로그
if (Array.isArray(videoUrls)) { // videoUrls가 배열인지 확인
const gallery = document.getElementById('video-gallery');
videoUrls.forEach((url, index) => {
const thumbnail = document.createElement('div');
thumbnail.className = 'video-thumbnail';
thumbnail.innerText = `Video ${index + 1}`;
thumbnail.addEventListener('click', () => playVideo(url));
gallery.appendChild(thumbnail);
});
} else {
console.error('Fetched data is not an array:', videoUrls);
}
})
.catch(error => {
console.error('Error fetching video URLs:', error); // 에러 디버깅용 콘솔 로그
});
// 비디오 재생 함수
function playVideo(url) {
const videoPlayer = document.getElementById('video-player');
const video = document.getElementById('video');
const videoSource = document.getElementById('video-source');
videoSource.src = url;
video.load();
videoPlayer.style.display = 'block';
}
</script>
그 다음 js적인 요소 설명이다.
우선 fetch 방식으로 비동기적으로 서버와 통신을 한다.
서버와 통신에 성공하면, 서버에서는 urls (Storage 링크 배열)을 넘겨 준다. ( 위에 내용 참고)
if문에서는 배열이 맞는지를 확인하고,
배열이 맞으면, 그 크기만큼 반복문을 돌리고 thumnail을 나타낼 div 요소를 만들어준다.
그리고 class와 innerText 값을 주어주고, 만일 thumnail 이 클릭 된 경우 playVideo 함수를 실행하고,
video-gallery에 해당 요소를 넣어준다.
playVideo 함수는 url값을 전달 받고, vidooSource src에 해당 url을 넣어주고, 비디오를 로드하여 재생한다.
(해당 video-player 요소는 display가 none에서 block 처리됨)
📌 실행 결과
이제 내가 만들었던 flask 서버에 접속하고 썸네일을 누르면,
해당 영상이 웹페이지 내에서 잘 재생되고 있는 것을 확인 할 수 있다 .
우선 기능은 완성되었고, 디자인은 css로 쿰쩍쿰쩍 잘 하면 될 것이다...!
'Backend > SQL' 카테고리의 다른 글
SQLD 필수 개념 요약 정리 / 벼락치기, 독학, 요약본 (2) | 2024.08.21 |
---|---|
RDBMS NoSQL MongoDB (0) | 2024.07.08 |
[Firebase] 데이터베이스에 CSV 파일 Import (0) | 2024.05.21 |
[Firebase] "error" : "Permission denied" 오류 (0) | 2024.05.20 |