-
Notifications
You must be signed in to change notification settings - Fork 795
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Screen streaming under docker environment (#674)
Co-authored-by: Shuchang Zheng <[email protected]>
- Loading branch information
1 parent
9342dfb
commit 3f92c50
Showing
12 changed files
with
222 additions
and
10 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,38 @@ | ||
import asyncio | ||
import subprocess | ||
|
||
import structlog | ||
import typer | ||
|
||
from skyvern.forge import app | ||
from skyvern.forge.sdk.settings_manager import SettingsManager | ||
|
||
INTERVAL = 1 | ||
LOG = structlog.get_logger() | ||
|
||
|
||
async def run() -> None: | ||
file_name = "skyvern_screenshot.png" | ||
png_file_path = f"{SettingsManager.get_settings().STREAMING_FILE_BASE_PATH}/{file_name}" | ||
|
||
while True: | ||
# run subprocess to take screenshot | ||
subprocess.run( | ||
f"xwd -root | xwdtopnm 2>/dev/null | pnmtopng > {png_file_path}", shell=True, env={"DISPLAY": ":99"} | ||
) | ||
|
||
# upload screenshot to S3 | ||
try: | ||
await app.STORAGE.save_streaming_file("placeholder_org", file_name) | ||
except Exception: | ||
LOG.info("Failed to save screenshot") | ||
|
||
await asyncio.sleep(INTERVAL) | ||
|
||
|
||
def main() -> None: | ||
asyncio.run(run()) | ||
|
||
|
||
if __name__ == "__main__": | ||
typer.run(main) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
#!/bin/bash | ||
|
||
echo "Starting streaming..." | ||
|
||
while true; do | ||
xwd -root | xwdtopnm | pnmtopng > /tmp/skyvern_screenshot.png | ||
sleep 1 | ||
done |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,115 @@ | ||
import asyncio | ||
import base64 | ||
from datetime import datetime | ||
|
||
import structlog | ||
from fastapi import APIRouter, WebSocket, WebSocketDisconnect | ||
from pydantic import ValidationError | ||
from websockets.exceptions import ConnectionClosedOK | ||
|
||
from skyvern.forge import app | ||
from skyvern.forge.sdk.schemas.tasks import TaskStatus | ||
from skyvern.forge.sdk.services.org_auth_service import get_current_org | ||
|
||
LOG = structlog.get_logger() | ||
websocket_router = APIRouter() | ||
STREAMING_TIMEOUT = 300 | ||
|
||
|
||
@websocket_router.websocket("/tasks/{task_id}") | ||
async def task_stream( | ||
websocket: WebSocket, | ||
task_id: str, | ||
apikey: str | None = None, | ||
token: str | None = None, | ||
) -> None: | ||
try: | ||
await websocket.accept() | ||
if not token and not apikey: | ||
await websocket.send_text("No valid credential provided") | ||
return | ||
except ConnectionClosedOK: | ||
LOG.info("ConnectionClosedOK error. Streaming won't start") | ||
return | ||
|
||
try: | ||
organization = await get_current_org(x_api_key=apikey, authorization=token) | ||
organization_id = organization.organization_id | ||
except Exception: | ||
try: | ||
await websocket.send_text("Invalid credential provided") | ||
except ConnectionClosedOK: | ||
LOG.info("ConnectionClosedOK error while sending invalid credential message") | ||
return | ||
|
||
LOG.info("Started task streaming", task_id=task_id, organization_id=organization_id) | ||
# timestamp last time when streaming activity happens | ||
last_activity_timestamp = datetime.utcnow() | ||
|
||
try: | ||
while True: | ||
# if no activity for 5 minutes, close the connection | ||
if (datetime.utcnow() - last_activity_timestamp).total_seconds() > STREAMING_TIMEOUT: | ||
LOG.info( | ||
"No activity for 5 minutes. Closing connection", task_id=task_id, organization_id=organization_id | ||
) | ||
await websocket.send_json( | ||
{ | ||
"task_id": task_id, | ||
"status": "timeout", | ||
} | ||
) | ||
return | ||
|
||
task = await app.DATABASE.get_task(task_id=task_id, organization_id=organization_id) | ||
if not task: | ||
LOG.info("Task not found. Closing connection", task_id=task_id, organization_id=organization_id) | ||
await websocket.send_json( | ||
{ | ||
"task_id": task_id, | ||
"status": "not_found", | ||
} | ||
) | ||
return | ||
if task.status.is_final(): | ||
LOG.info( | ||
"Task is in a final state. Closing connection", | ||
task_status=task.status, | ||
task_id=task_id, | ||
organization_id=organization_id, | ||
) | ||
await websocket.send_json( | ||
{ | ||
"task_id": task_id, | ||
"status": task.status, | ||
} | ||
) | ||
return | ||
|
||
if task.status == TaskStatus.running: | ||
file_name = f"{task_id}.png" | ||
screenshot = await app.STORAGE.get_streaming_file(organization_id, file_name) | ||
if screenshot: | ||
encoded_screenshot = base64.b64encode(screenshot).decode("utf-8") | ||
await websocket.send_json( | ||
{ | ||
"task_id": task_id, | ||
"status": task.status, | ||
"screenshot": encoded_screenshot, | ||
} | ||
) | ||
last_activity_timestamp = datetime.utcnow() | ||
await asyncio.sleep(2) | ||
|
||
except ValidationError as e: | ||
await websocket.send_text(f"Invalid data: {e}") | ||
except WebSocketDisconnect: | ||
LOG.info("WebSocket connection closed") | ||
except ConnectionClosedOK: | ||
LOG.info("ConnectionClosedOK error while streaming", exc_info=True) | ||
return | ||
except Exception: | ||
LOG.warning("Error while streaming", exc_info=True) | ||
return | ||
LOG.info("WebSocket connection closed successfully") | ||
return |