2024-06-01 13:24:40 +10:00
|
|
|
"""
|
|
|
|
This module provides a FastAPI application that uses Playwright to fetch and return
|
|
|
|
the HTML content of a specified URL. It supports optional proxy settings and media blocking.
|
|
|
|
"""
|
|
|
|
|
|
|
|
from os import environ
|
|
|
|
|
2024-04-25 10:31:28 -07:00
|
|
|
from fastapi import FastAPI
|
2024-04-15 17:01:47 -04:00
|
|
|
from fastapi.responses import JSONResponse
|
2024-06-01 13:24:40 +10:00
|
|
|
from playwright.async_api import Browser, async_playwright
|
2024-04-15 17:01:47 -04:00
|
|
|
from pydantic import BaseModel
|
2024-06-13 17:08:40 -03:00
|
|
|
from get_error import get_error
|
2024-05-24 17:41:34 +02:00
|
|
|
|
2024-05-31 15:39:54 -07:00
|
|
|
PROXY_SERVER = environ.get("PROXY_SERVER", None)
|
|
|
|
PROXY_USERNAME = environ.get("PROXY_USERNAME", None)
|
|
|
|
PROXY_PASSWORD = environ.get("PROXY_PASSWORD", None)
|
|
|
|
BLOCK_MEDIA = environ.get("BLOCK_MEDIA", "False").upper() == "TRUE"
|
2024-04-25 10:31:28 -07:00
|
|
|
|
2024-04-15 17:01:47 -04:00
|
|
|
app = FastAPI()
|
|
|
|
|
|
|
|
class UrlModel(BaseModel):
|
2024-06-01 13:24:40 +10:00
|
|
|
"""Model representing the URL and associated parameters for the request."""
|
2024-04-15 17:01:47 -04:00
|
|
|
url: str
|
2024-06-01 13:46:16 +10:00
|
|
|
wait_after_load: int = 0
|
2024-06-01 13:10:14 +10:00
|
|
|
timeout: int = 15000
|
2024-05-31 15:39:54 -07:00
|
|
|
headers: dict = None
|
2024-04-15 17:01:47 -04:00
|
|
|
|
2024-04-25 10:31:28 -07:00
|
|
|
browser: Browser = None
|
|
|
|
|
|
|
|
@app.on_event("startup")
|
|
|
|
async def startup_event():
|
2024-06-01 13:24:40 +10:00
|
|
|
"""Event handler for application startup to initialize the browser."""
|
2024-04-25 10:31:28 -07:00
|
|
|
global browser
|
|
|
|
playwright = await async_playwright().start()
|
|
|
|
browser = await playwright.chromium.launch()
|
|
|
|
|
|
|
|
@app.on_event("shutdown")
|
|
|
|
async def shutdown_event():
|
2024-06-01 13:24:40 +10:00
|
|
|
"""Event handler for application shutdown to close the browser."""
|
2024-04-25 10:31:28 -07:00
|
|
|
await browser.close()
|
2024-04-15 17:01:47 -04:00
|
|
|
|
2024-04-25 10:31:28 -07:00
|
|
|
@app.post("/html")
|
|
|
|
async def root(body: UrlModel):
|
2024-06-01 13:24:40 +10:00
|
|
|
"""
|
|
|
|
Endpoint to fetch and return HTML content of a given URL.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
body (UrlModel): The URL model containing the target URL, wait time, and timeout.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
JSONResponse: The HTML content of the page.
|
|
|
|
"""
|
2024-05-24 17:41:34 +02:00
|
|
|
context = None
|
|
|
|
if PROXY_SERVER and PROXY_USERNAME and PROXY_PASSWORD:
|
2024-05-31 15:39:54 -07:00
|
|
|
context = await browser.new_context(
|
|
|
|
proxy={
|
|
|
|
"server": PROXY_SERVER,
|
|
|
|
"username": PROXY_USERNAME,
|
|
|
|
"password": PROXY_PASSWORD,
|
|
|
|
}
|
|
|
|
)
|
2024-05-24 17:41:34 +02:00
|
|
|
else:
|
|
|
|
context = await browser.new_context()
|
|
|
|
|
|
|
|
if BLOCK_MEDIA:
|
2024-05-31 15:39:54 -07:00
|
|
|
await context.route(
|
|
|
|
"**/*.{png,jpg,jpeg,gif,svg,mp3,mp4,avi,flac,ogg,wav,webm}",
|
|
|
|
handler=lambda route, request: route.abort(),
|
|
|
|
)
|
2024-05-24 17:41:34 +02:00
|
|
|
|
2024-04-25 10:31:28 -07:00
|
|
|
page = await context.new_page()
|
2024-05-31 15:39:54 -07:00
|
|
|
|
|
|
|
# Set headers if provided
|
|
|
|
if body.headers:
|
|
|
|
await page.set_extra_http_headers(body.headers)
|
|
|
|
|
2024-06-13 17:08:40 -03:00
|
|
|
response = await page.goto(
|
2024-05-21 14:53:57 +08:00
|
|
|
body.url,
|
|
|
|
wait_until="load",
|
2024-06-01 13:10:14 +10:00
|
|
|
timeout=body.timeout,
|
2024-05-21 14:53:57 +08:00
|
|
|
)
|
2024-06-13 17:08:40 -03:00
|
|
|
page_status_code = response.status
|
|
|
|
page_error = get_error(page_status_code)
|
2024-05-22 10:45:43 -07:00
|
|
|
# Wait != timeout. Wait is the time to wait after the page is loaded - useful in some cases were "load" / "networkidle" is not enough
|
2024-06-01 13:46:16 +10:00
|
|
|
if body.wait_after_load > 0:
|
|
|
|
await page.wait_for_timeout(body.wait_after_load)
|
2024-06-01 13:24:40 +10:00
|
|
|
|
2024-04-25 10:31:28 -07:00
|
|
|
page_content = await page.content()
|
|
|
|
await context.close()
|
2024-06-13 17:08:40 -03:00
|
|
|
json_compatible_item_data = {
|
|
|
|
"content": page_content,
|
|
|
|
"pageStatusCode": page_status_code,
|
|
|
|
"pageError": page_error
|
|
|
|
}
|
|
|
|
return JSONResponse(content=json_compatible_item_data)
|