-
Notifications
You must be signed in to change notification settings - Fork 9
New issue
Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? # to your account
Add a ProxyResponse class to fastapi_proxy_lib #15
Comments
Thank you very much for your valuable advice. Can you provide more specific use cases? For example, what specific functionality do you intend to implement using this feature? Can It seems like you've read the source code of |
用例差不多就这些,大差不差就行,fastapi_proxy_lib.core.http的话文档不全,具体也没怎么看,建议先把文档完善了 |
代理这东西最重要的估计就是鉴权和双向修改,那个示例啥都没有,与其那样还不如上nginx或者302 |
你提到的这些算是高级用例了,要配合httpx来弄。 如果要鉴权的话,你看看这个,https://www.python-httpx.org/advanced/#customizing-authentication
比如 class MyCustomAuth(httpx.Auth):
def __init__(self, token):
self.token = token
def auth_flow(self, request):
# Send the request, with a custom `X-Authentication` header.
request.headers['X-Authentication'] = self.token
yield request
reverse_http_app(client=httpx.AsyncClient(auth=MyCustomAuth("mytoken")), base_url=base_url) 双向修改的话,对于请求和响应的headers(包括cookies)的修改我都能理解,但是修改body似乎是一个很奇怪的用例。我不知道这样做的实际业务场景是什么。 如果你希望弄 给他传一个自定义transports进去就行。 import httpx
class MyTransport(httpx.AsyncHTTPTransport):
async def handle_async_request(self, request):
request.headers["foo"] = "bar"
target_resp = await super().handle_async_request(request)
target_resp.headers["a"] = "b"
return target_resp
reverse_http_app(client=httpx.AsyncClient(transport=MyTransport()), base_url=base_url) 非常不建议修改httpx.response中涉及body的部分:
你提到的nginx或者302,和这个项目的应用场景不太一样。
文档的话,我英文不太行,所以也就不太想弄高级部分的文档。 欢迎对于文档的PR,我会合并它。 Welcome PR about docs ! |
ProxyResponse还是有必要加一个 |
对于单独写类实现的话感觉相对来说会麻烦很多,有些时候也就是简单改个headers,或者对返回的herders做一些改动,而你这个只能改client,还保不准以后httpx还更不更,不更的话引用你的库的那些又得大面积重构 |
'''
''' |
这个库在设计的时候就是把对 就算发生了你说的httpx更改API的情况, pip install fastapi-proxy-lib httpx==0.25 我可以提供一个 但是body的话,可能很难设计出一个API来满足大多数需求。 |
我还是不太明白你最开始要求的 代理鉴权确实是个很常用的场景, 但是大步幅修改 |
看了一下,改body倒是可以用httpx.AsyncHTTPTransport(最好写个helper类套一层,免得改api),毕竟不常用,文档注明就行,前后改headers和自定义鉴权还是有必要,就比如说代理bilibili视频下载url,最方便的就是像写api那样直接鉴权前端并返回ProxyResponse同时更改headers |
我想了一下,能实现你的需求的妥协方法,用fastapi_proxy_lib.core.http.BaseHttpProxy 如果要修改请求头或者请求体的话,就用我前面提到的 如果你想修改响应头,代码,响应体这些,你把最终返回的 不建议使用 Example: import httpx
from fastapi import FastAPI, Request
from fastapi.responses import StreamingResponse
from fastapi_proxy_lib.core.http import BaseHttpProxy
class MyCustomAuth(httpx.Auth):
def auth_flow(self, request: httpx.Request):
request.headers.update({
"referer": "https://www.bilibili.com/video/",
"user-agent": "My User Agent 1.0",
})
yield request
proxy = BaseHttpProxy(httpx.AsyncClient(auth=MyCustomAuth()))
app = FastAPI()
@app.get("/proxy/{vid}/{video}.mp4")
async def _(request: Request):
target_url = httpx.URL(get_bili_url(vid, video))
target_resp = await proxy.send_request_to_target(request=request, target_url=target_url)
if isinstance(target_resp, StreamingResponse):
# do some processing, whatever you want
new_resp = StreamingResponse(
content=target_resp.body_iterator,
status_code=target_resp.status_code,
headers=target_resp.headers,
media_type=target_resp.media_type,
)
else:
new_resp = target_resp
return new_resp 很难提供一个
至于你提到的包装一层,免得修改API: |
改的话主要也就是前后改,body可以直接拿类来操作,就比如写个helper类,要改的话写个类继承,然后改完后再用super调用请求方法,返回后再改,而如果只是在前面改的话加个headers和body参数就行,拿到之后直接update,ProxyResponse倒是可以继承StreamingResponse,然后改写初始化方法,还有代理文件传输url时概率出现WARNING:asyncio:socket.send() raised exception.,最好看看会不会有问题 |
抱歉回复晚了,前几天有点累。
我是感觉,即使我提供了 重要的是我写个文档来说明这些。 至于 @app.get("/proxy/{vid}/{video}.mp4")
async def _(request: Request):
target_url = httpx.URL(get_bili_url(vid, video))
target_resp = await proxy.send_request_to_target(request=request, target_url=target_url)
if isinstance(target_resp, StreamingResponse):
# do some processing, whatever you want
new_resp = StreamingResponse(
content=target_resp.body_iterator,
status_code=target_resp.status_code,
headers=target_resp.headers,
media_type=target_resp.media_type,
) 直接继承 你最开始的要求,完全可以用现有的功能来实现(就是我上面给出的例子),而且灵活性还更好,也就多了几行代码而已。 引入新的东西就要增加更多维护成本、测试。
这个是有测试的,如果你发现代理bilibili那边有问题,再发一个issue,我会研究下。 fastapi-proxy-lib/tests/test_http.py Lines 180 to 185 in ea58f66
|
主要是麻烦,而且之前总有些奇奇怪怪的bug,如WARNING:asyncio:socket.send() raised exception.刷屏,不处理好的话分分钟服务器就得蹦 |
最好还是专门写一个ProxyResponse来处理,一是在需求没那么复杂的情况下减少一定的代码量,二是避免一些乱七八糟的bug |
初始化参数的话就request target_url transport应该就差不多了,后面 headers属性和status_code再做成可更改的,简单的直接就改requests和后面response就行,复杂的定义transport也能解决 |
对于transport最好导出一个helper类来继承,免得后面改api |
core.zip |
为回复晚了感到抱歉。 不要直接发文件,最好按照贡献指南来操作。 zip里面我大概看了下,有涉及底层ASGI协议的部分。我不喜欢处理底层的细节,这样维护起来很麻烦。 而且还缺少测试。 在 PR #22 里面,我把 最开始你提到的 这个库设计的时候就十分重视代理的透明性(无损转发),修改应该保持到最低限度(如代理鉴权等)。 如果没有新的问题的话,我建议关闭这个issue;如果有别人也有类似的需求,我会重新打开issue。 |
按计划关闭,如有新的需求或者想法再重新打开吧。 |
ProxyResponse还是建议加进去,那个zip说实在的也就加了个ProxyResponse,其他没改 |
I would like to request a new feature for fastapi_proxy_lib, which is a library that allows fastapi users to easily create proxy endpoints. The feature is a ProxyResponse class that can be used as a return value for fastapi routes. The ProxyResponse class would take care of sending a request to a target URL, receiving the response, and forwarding it to the client. The ProxyResponse class would also allow the user to customize the request and response headers, as well as the response content, by passing optional arguments or a callback function.
Example
Here is an example of how the ProxyResponse class could be used in a fastapi app:
In this example, the
/foo
endpoint would proxy the request tohttp://www.example.com/foo/
, using theGET
method and the customUser-Agent
header. The response would be forwarded to the client, with theContent-Type
header set totext/plain
. The response content would also be processed by themy_respfun
function, which could modify the headers, status code, or content as needed.The ProxyResponse class would have the following constructor parameters:
url
: The target URL to proxy the request to. Required.method
: The HTTP method to use for the proxy request. Optional. Default: The same method as the original request.params
: The query parameters to use for the proxy request. Optional. Default: The same parameters as the original request.data
: The request body to use for the proxy request. Optional. Default: The same body as the original request.json
: The JSON data to use for the proxy request. Optional. Default: None. If provided, it will override thedata
parameter and set theContent-Type
header toapplication/json
.files
: The files to use for the proxy request. Optional. Default: None. If provided, it will override thedata
parameter and set theContent-Type
header tomultipart/form-data
.reqheaders
: A dictionary of custom headers to use for the proxy request. Optional. Default: The same headers as the original request, except for theHost
header, which will be set to the target URL's host.respheaders
: A dictionary of custom headers to use for the proxy response. Optional. Default: The same headers as the target response, except for theTransfer-Encoding
header, which will be removed if the response is not streaming.respfun
: A callback function to process the response headers, status code, and content. Optional. Default: None. If provided, it should take three arguments:headers
,status_code
, andcontent
, and return a generator that yields a dictionary with the keysheaders
andstatus_code
, followed by the modified content. Theheaders
andstatus_code
values will be used for the proxy response, and the content will be streamed to the client.The text was updated successfully, but these errors were encountered: