9

我正在尝试提供 package_docs 目录中的静态文件。当我在浏览器中打开时:

http://127.0.0.1:8001/packages/docs/index.html,页面正在运行。

但我想打开页面: http: //127.0.0.1 :8001/packages/docs/

没有源文件。输出是 404 Not Found

app.mount("/packages/docs", 
    StaticFiles(directory=pkg_resources.resource_filename(__name__, 'package_docs')
    ), 
    name="package_docs")

@app.get("/packages/docs/.*", include_in_schema=False)
def root():
    return HTMLResponse(pkg_resources.resource_string(__name__, "package_docs/index.html"))


app.include_router(static.router)
app.include_router(jamcam.router, prefix="/api/v1/cams", tags=["jamcam"])

如何更改我的代码?任何建议都会有所帮助。先感谢您。

4

4 回答 4

15

Starlette 中有一个 html 选项可以在 FastAPI 中使用。星光文档

这会让你有一些东西,比如:

app.mount("/site", StaticFiles(directory="site", html = True), name="site")

这会将 /site 解析为 /site/index.html,将 /site/foo/ 解析为 /site/foo/index.html 等。

如果您想以不使用“directory = /foo”处理的方式更改文件夹名称,其他答案可以帮助您重定向,但如果您只想加载关联的 .html 文件,这是最简单的选项。

于 2020-09-09T05:55:47.890 回答
8

您需要使用 FastAPI TemplateResponse(实际上是 Starlette):

from fastapi import Request
from fastapi.staticfiles import StaticFiles
from fastapi.templating import Jinja2Templates

app.mount("/static", StaticFiles(directory="static"), name="static")

templates = Jinja2Templates(directory="package_docs")

@app.get("/items/{id}")
async def example(request: Request):
    return templates.TemplateResponse("index.html", {"request": request})

Request作为 Jinja2 上下文中键值对的一部分。因此,您还必须将其声明为查询参数。并且你必须指定一个你想用 Jinja 渲染它的 html 文件("your.html", {"request": request})

还可以直接返回 HTMLResponse,您可以使用HTMLResponsefromfastapi.responses

from fastapi.responses import HTMLResponse

@app.get("/items/", response_class=HTMLResponse)
async def read_items():
    return """
    <html>
        <head>
            <title></title>
        </head>
        <body>
        </body>
    </html>
    """

您可以从FastAPI 自定义响应中阅读有关自定义响应的更多信息

于 2020-07-16T16:33:54.333 回答
0

文档

第一个“/static”指的是这个“子应用程序”将被“挂载”到的子路径。因此,任何以“/static”开头的路径都将由它处理。

这意味着您将目录安装在http://127.0.0.1:8001/packages/docs/,但是您需要在 URL 中指定一个文件,或者像您一样添加一个处理程序。但是问题在于,由于您首先安装了路径,因此它不会考虑包含部分路径的以下路径。

一种可能性是首先指定http://127.0.0.1:8001/packages/docs/的路径,以便它由 fastapi 处理,然后挂载文件夹,提供静态文件。

另外,我会将请求http://127.0.0.1:8001/packages/docs/的用户重定向到http://127.0.0.1:8001/packages/docs/index.html

于 2020-06-18T20:08:02.403 回答
-1

如果app.mount不是一个选项,您始终可以手动读取文件并使用其内容进行响应...

例如,如果您的静态文件在里面,/site那么:

from os.path import isfile
from fastapi import Response
from mimetypes import guess_type


@app.get("/site/{filename}")
async def get_site(filename):
    filename = './site/' + filename

    if not isfile(filename):
        return Response(status_code=404)

    with open(filename) as f:
        content = f.read()

    content_type, _ = guess_type(filename)
    return Response(content, media_type=content_type)


@app.get("/site/")
async def get_site_default_filename():
    return await get_site('index.html')
于 2020-12-17T22:28:22.630 回答