0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Fast API x Reactで動画保存システムの構築

Posted at

概要

こちらの記事はReact x Fast APIで動画保存システムを作ってみたという内容になります
これまで記事を読んでくださった方は分かるかもしれませんが、今まで学んだことを形にしてみようという感じです

バージョン情報

yt-dlp==2024.12.6
fastapi==0.115.6
uvicorn==0.32.1

"axios": "^1.7.9",
"react": "^19.0.0",
"react-dom": "^19.0.0"
"typescript": "~5.7.2",

イメージ図

構成は至ってシンプルな形です
React,TypeScriptで画面を作り、裏側のFast APIで作ったエンドポイントを呼び出すといった流れをとります

名称未設定ファイル.drawio.png

(本当はバックエンドからファイルを返す形にしようとしていましたが、ちょっと間に合わず、、)

実装内容

フロント

cssなどもありますが、一旦実際のロジック部分を記載します

import React, { useState } from 'react';
import axios from 'axios';
import './App.css';  // CSSをインポート

const VideoDownloader = () => {
  const [targetLink, setUrl] = useState('');
  const [message, setMessage] = useState('');

  const handleSubmit = async (event: React.FormEvent) => {
    event.preventDefault();
    try {
      // FastAPIエンドポイントにPOSTリクエストを送信
      const response = await axios.post(
        'http://localhost:xxxx/yt_download',
        { url: targetLink },
        {
          headers: {
            'Content-Type': 'application/json',
          },
        },
      );

      if (response.status === 200) {
        setMessage('動画のダウンロードが完了しました!');
      } else {
        setMessage('動画のダウンロードが完了していません、再度お試しください。');
      }

    } catch (error: any) {
      // エラーハンドリング
      setMessage(`エラー: ${error.response.status} - ${error.response.data.detail}`);
    }
  };

  return (
    <div className="container">
      <h1>動画URLを入力してください</h1>
      <form onSubmit={handleSubmit}>
        <input
          type="text"
          value={targetLink}
          onChange={(e) => setUrl(e.target.value)}
          placeholder="動画のURL"
        />
        <button type="submit">ダウンロード</button>
      </form>
      {message && <p>{message}</p>}
    </div>
  );
};

export default VideoDownloader;
  • export default VideoDownloader;でコンポーネントとして定義し、他ファイルでもインポートできるような形にしております
  • axiosというものを使用して、バックエンドに向けてリクエストを送っています
  • returnの部分でダウンロード前後の画面表示を行っています
    • ダウンロードの処理が成功失敗によって、メッセージを渡してreturn側で処理するといった感じです

バックエンド

こちらは以前紹介した記事と大きくは変わっていません
ただ、2点だけ変更を加えております

1点目がCORSエラー対策です

これは「同一オリジンポリシー(SOP)の制限を回避する際に利用する機能のこと。異なるオリジンからのアクセスを許可できる仕組み」のようです
正直いってまだまだ理解が浅いのでなんとも言えません

いろいろ情報をみて理解した範囲ではhttp://localhost同士で通信するとエラーが出るぞといった感覚です
そのため、バックエンドではCORSエラーを回避する設定が必要になります

Fast APIを使っている場合はfastapi.middleware.corsからCORSMiddlewareを使用してどういうoriginを許可するかどのリクエストメソッドを許可するかなどを設定することができます

以下は使用例でhttp://localhost:xxxxを許可する、リクエストメソッドはなんでも良いといった感じですね

from fastapi.middleware.cors import CORSMiddleware

origins = ["http://localhost:xxxx"]
app.add_middleware(
    CORSMiddleware,
    allow_origins=origins,
    allow_credentials=True,
    allow_methods=["*"],
    allow_headers=["*"],
)

2点目が動画ダウンロードのメソッド内でURLでないものが渡された場合に、400エラーをとメッセージを返すように変更しております
(if re.matchの部分はwithよりも外でも良いような気がします)

@app.post("/yt_download")
def download_yt_video(jsonbody: get_yt):
    try:
        download_link = jsonbody.url
        with YoutubeDL(ytdlp_opts) as ydl:
            if re.match(URL_PATTERN, download_link):
                ydl.download(download_link)
            else:
                raise HTTPException(status_code=400)
    except HTTPException:
        raise HTTPException(
            detail="不正なURLです。URLを入力してください。"
        )
    except Exception:
        raise HTTPException(status_code=500, detail="予期せぬエラーが発生しました。")

完成した画面

  • 初期画面
    スクリーンショット 2025-03-01 23.31.01.png

  • ダウンロード成功
    スクリーンショット 2025-03-01 23.42.15.png

  • ダウンロード失敗
    スクリーンショット 2025-03-01 23.55.05.png

おわりに

ということで、フロントとバックエンドを作ってみたという回でした

今後もまた何か作ってみたいと思います!

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?