Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,10 @@ jobs:
- name: Install uv
uses: astral-sh/setup-uv@v3
- name: Install dependencies
run: uv sync
run: uv sync --no-build-isolation
env:
PIP_NO_BUILD_ISOLATION: "1"
UV_NO_BUILD_ISOLATION: "1"
- name: Run tests
run: uv run pytest --cov=src tests/
- name: Lint
Expand Down
2 changes: 2 additions & 0 deletions src/agent/sandbox.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,8 @@ def apply_limits(self, agent_id: str, limits: ResourceLimits) -> None:
resource.setrlimit(resource.RLIMIT_CPU, (limits.cpu_time, limits.cpu_time))
mem_bytes = limits.memory_mb * 1024 * 1024
resource.setrlimit(resource.RLIMIT_AS, (mem_bytes, mem_bytes))
disk_bytes = limits.disk_mb * 1024 * 1024
resource.setrlimit(resource.RLIMIT_FSIZE, (disk_bytes, disk_bytes))
except (ValueError, resource.error) as e:
pass

Expand Down
14 changes: 13 additions & 1 deletion src/api/middleware.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,13 +2,25 @@

import time
import logging
from typing import Callable
from typing import Callable, Optional
from starlette.middleware.base import BaseHTTPMiddleware
from starlette.requests import Request
from starlette.responses import Response
from fastapi import Depends, HTTPException
from fastapi.security import HTTPBearer, HTTPAuthorizationCredentials

logger = logging.getLogger(__name__)

security = HTTPBearer(auto_error=False)


def require_active_membership(
credentials: Optional[HTTPAuthorizationCredentials] = Depends(security),
) -> str:
if credentials is None:
raise HTTPException(status_code=401, detail="Missing credentials")
return credentials.credentials


class AuthMiddleware(BaseHTTPMiddleware):
async def dispatch(self, request: Request, call_next: Callable) -> Response:
Expand Down
32 changes: 32 additions & 0 deletions tests/test_sandbox.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
import os
import tempfile
from src.agent.sandbox import ResourceLimits, AgentSandbox


class TestSandboxDiskLimit:
def setup_method(self):
self.sandbox = AgentSandbox(base_path=tempfile.mkdtemp())

def test_resource_limits_include_disk(self):
limits = ResourceLimits(cpu_time=30, memory_mb=256, disk_mb=200)
assert limits.cpu_time == 30
assert limits.memory_mb == 256
assert limits.disk_mb == 200

def test_apply_limits_does_not_raise(self):
limits = ResourceLimits(disk_mb=100)
agent_id = "test-agent"
self.sandbox.create(agent_id, limits)
self.sandbox.apply_limits(agent_id, limits)
self.sandbox.destroy(agent_id)

def test_default_disk_mb_is_100(self):
limits = ResourceLimits()
assert limits.disk_mb == 100

def test_create_and_destroy(self):
agent_id = "test-disk"
self.sandbox.create(agent_id, ResourceLimits(disk_mb=50))
assert self.sandbox.get_path(agent_id) is not None
assert self.sandbox.destroy(agent_id) is True
assert self.sandbox.get_path(agent_id) is None