为使用 Flask 的 create-react-app 创建的前端提供服务

IT技术 python reactjs flask create-react-app
2021-04-04 15:52:09

我有一个带有 API 路由的 Flask 后端,这些路由由使用 create-react-app 创建的 React 单页应用程序访问。使用 create-react-app 开发服务器时,我的 Flask 后端工作正常。

我想npm run build从我的 Flask 服务器提供构建(使用)静态 React 应用程序。构建 React 应用程序会导致以下目录结构:

- build
  - static
    - css
        - style.[crypto].css
        - style.[crypto].css.map
    - js
        - main.[crypto].js
        - main.[crypto].js.map
  - index.html
  - service-worker.js
  - [more meta files]

通过[crypto],我的意思是在构建时所产生的随机生成的字符串。

收到index.html文件后,浏览器会发出以下请求:

- GET /static/css/main.[crypto].css
- GET /static/css/main.[crypto].css
- GET /service-worker.js

我应该如何提供这些文件?我想出了这个:

from flask import Blueprint, send_from_directory

static = Blueprint('static', __name__)

@static.route('/')
def serve_static_index():
    return send_from_directory('../client/build/', 'index.html')

@static.route('/static/<path:path>') # serve whatever the client requested in the static folder
def serve_static(path):
    return send_from_directory('../client/build/static/', path)

@static.route('/service-worker.js')
def serve_worker():
    return send_from_directory('../client/build/', 'service-worker.js')

这样,静态资产就成功服务了。

另一方面,我可以将它与内置的 Flask 静态实用程序结合起来。但我不明白如何配置它。

我的解决方案足够健壮吗?有没有办法使用内置的 Flask 功能来服务这些资产?有没有更好的方法来使用 create-react-app?

5个回答
import os
from flask import Flask, send_from_directory

app = Flask(__name__, static_folder='react_app/build')

# Serve React App
@app.route('/', defaults={'path': ''})
@app.route('/<path:path>')
def serve(path):
    if path != "" and os.path.exists(app.static_folder + '/' + path):
        return send_from_directory(app.static_folder, path)
    else:
        return send_from_directory(app.static_folder, 'index.html')


if __name__ == '__main__':
    app.run(use_reloader=True, port=5000, threaded=True)

这就是我的结局。所以基本上捕获所有路由,测试路径是否为文件 => 发送文件 => 否则发送 index.html。这样您就可以从您希望的任何路线重新加载 react 应用程序,并且不会中断。

@ user3216673 很可能不是flask 或create-react-app 的问题,而是您的浏览器的问题。随意猜测:取消注册您的 Service Worker 可能会解决该问题。
2021-05-25 15:52:09
我在使用此解决方案时遇到 mime 类型错误:-( The script has an unsupported MIME type ('text/html'). /service-worker.js Failed to load resource: net::ERR_INSECURE_RESPONSE registerServiceWorker.js:71 Error during service worker registration: DOMException: Failed to register a ServiceWorker: The script has an unsupported MIME type ('text/html').
2021-06-09 15:52:09
您可以使用 app.static_folder 来保持代码干燥
2021-06-17 15:52:09

首先做npm run build上面提到的构建静态生产文件

from flask import Flask, render_template

app = Flask(__name__, static_folder="build/static", template_folder="build")

@app.route("/")
def hello():
    return render_template('index.html')

print('Starting Flask!')

app.debug=True
app.run(host='0.0.0.0')

不幸的是,我认为您无法通过开发热重载来实现它。

这里有一个可行的解决方案。有没有想过为什么我们需要两个单独的文件夹statictemplates. 为了隔离混乱,对吧?但是,它与生产版本的问题,因为它有两个一个文件夹statictemplates类型的文件和所有相关性链接这样。

build如果您同时考虑static和 ,则将提供文件夹templates

使用这样的东西

from flask import Flask, render_template

app = Flask(__name__, static_url_path='',
                  static_folder='build',
                  template_folder='build')

@app.route("/")
def hello():
    return render_template("index.html")

您的flask应用程序将运行良好。

这个答案救了我。所有其他人都有一些不适合我的设置的警告。
2021-06-14 15:52:09

接受的答案对我不起作用。我用过了

import os

from flask import Flask, send_from_directory, jsonify, render_template, request

from server.landing import landing as landing_bp
from server.api import api as api_bp

app = Flask(__name__, static_folder="../client/build")
app.register_blueprint(landing_bp, url_prefix="/landing")
app.register_blueprint(api_bp, url_prefix="/api/v1")


@app.route("/")
def serve():
    """serves React App"""
    return send_from_directory(app.static_folder, "index.html")


@app.route("/<path:path>")
def static_proxy(path):
    """static folder serve"""
    file_name = path.split("/")[-1]
    dir_name = os.path.join(app.static_folder, "/".join(path.split("/")[:-1]))
    return send_from_directory(dir_name, file_name)


@app.errorhandler(404)
def handle_404(e):
    if request.path.startswith("/api/"):
        return jsonify(message="Resource not found"), 404
    return send_from_directory(app.static_folder, "index.html")


@app.errorhandler(405)
def handle_405(e):
    if request.path.startswith("/api/"):
        return jsonify(message="Mehtod not allowed"), 405
    return e


我使用了一个只有一个路由的flask服务器/它从 Create react app(CRA) 的构建文件夹中读取 index.html 文件

from flask import Flask
app = Flask(__name__)
app.static_folder =  '../build'

@app.route('/')
def index():
    fref = open(r'../build/index.html')
    html_text = fref.read()
    fref.close()
    return html_text

app.run()

以这种方式设置我遇到了错误,由于路径不匹配,静态文件无法正确提供,所以我使用的解决方案是

  1. 在CRA的package.json中添加一个主页属性并设置为“/static”

{ "name":"App-name", "version":"", "dependencies":{} "homepage":"/static",....[其他键]}

 Add **homepage** key parallel to the **dependencies** key in the package.json file
  1. 这个主页属性将在CRA的构建过程中使用,用于代替index.html的%PUBLIC_URL%并附加到其他静态资产URL路径中(您可以在构建过程后查看index.html代码进行验证)

  2. 构建过程完成后,运行flask服务器,我们可以看到第一次带有/的GET请求index.html将被提供,然后是其他静态资产的请求/static/static/js/[[filename]]来自 HTML 文件,一切正常