mirror of
https://github.com/ggml-org/llama.cpp.git
synced 2026-06-25 22:00:21 +00:00
e27f308597
* server: avoid forwarding auth headers in CORS proxy * format * fix test * fix e2e test --------- Co-authored-by: Xuan Son Nguyen <son@huggingface.co>
182 lines
6.2 KiB
Python
182 lines
6.2 KiB
Python
import pytest
|
|
from openai import OpenAI
|
|
from utils import *
|
|
import threading
|
|
from http.server import BaseHTTPRequestHandler, ThreadingHTTPServer
|
|
|
|
server = ServerPreset.tinyllama2()
|
|
|
|
TEST_API_KEY = "sk-this-is-the-secret-key"
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def create_server():
|
|
global server
|
|
server = ServerPreset.tinyllama2()
|
|
server.api_key = TEST_API_KEY
|
|
|
|
|
|
@pytest.mark.parametrize("endpoint", ["/health", "/models"])
|
|
def test_access_public_endpoint(endpoint: str):
|
|
global server
|
|
server.start()
|
|
res = server.make_request("GET", endpoint)
|
|
assert res.status_code == 200
|
|
assert "error" not in res.body
|
|
|
|
|
|
def test_access_static_assets_without_api_key():
|
|
"""Static web UI assets should not require API key authentication (issue #21229)"""
|
|
global server
|
|
server.start()
|
|
for path in ["/", "/sw.js", "/manifest.webmanifest", "/_app/version.json"]:
|
|
res = server.make_request("GET", path)
|
|
assert res.status_code == 200, f"Expected 200 for {path}, got {res.status_code}"
|
|
|
|
|
|
@pytest.mark.parametrize("api_key", [None, "invalid-key"])
|
|
def test_incorrect_api_key(api_key: str):
|
|
global server
|
|
server.start()
|
|
res = server.make_request("POST", "/completions", data={
|
|
"prompt": "I believe the meaning of life is",
|
|
}, headers={
|
|
"Authorization": f"Bearer {api_key}" if api_key else None,
|
|
})
|
|
assert res.status_code == 401
|
|
assert "error" in res.body
|
|
assert res.body["error"]["type"] == "authentication_error"
|
|
|
|
|
|
def test_correct_api_key():
|
|
global server
|
|
server.start()
|
|
res = server.make_request("POST", "/completions", data={
|
|
"prompt": "I believe the meaning of life is",
|
|
}, headers={
|
|
"Authorization": f"Bearer {TEST_API_KEY}",
|
|
})
|
|
assert res.status_code == 200
|
|
assert "error" not in res.body
|
|
assert "content" in res.body
|
|
|
|
|
|
def test_correct_api_key_anthropic_header():
|
|
global server
|
|
server.start()
|
|
res = server.make_request("POST", "/completions", data={
|
|
"prompt": "I believe the meaning of life is",
|
|
}, headers={
|
|
"X-Api-Key": TEST_API_KEY,
|
|
})
|
|
assert res.status_code == 200
|
|
assert "error" not in res.body
|
|
assert "content" in res.body
|
|
|
|
|
|
def test_openai_library_correct_api_key():
|
|
global server
|
|
server.start()
|
|
client = OpenAI(api_key=TEST_API_KEY, base_url=f"http://{server.server_host}:{server.server_port}")
|
|
res = client.chat.completions.create(
|
|
model="gpt-3.5-turbo",
|
|
messages=[
|
|
{"role": "system", "content": "You are a chatbot."},
|
|
{"role": "user", "content": "What is the meaning of life?"},
|
|
],
|
|
)
|
|
assert len(res.choices) == 1
|
|
|
|
|
|
@pytest.mark.parametrize("origin,cors_header,cors_header_value", [
|
|
("localhost", "Access-Control-Allow-Origin", "localhost"),
|
|
("web.mydomain.fr", "Access-Control-Allow-Origin", "web.mydomain.fr"),
|
|
("origin", "Access-Control-Allow-Credentials", "true"),
|
|
("web.mydomain.fr", "Access-Control-Allow-Methods", "GET, POST"),
|
|
("web.mydomain.fr", "Access-Control-Allow-Headers", "*"),
|
|
])
|
|
def test_cors_options(origin: str, cors_header: str, cors_header_value: str):
|
|
global server
|
|
server.start()
|
|
res = server.make_request("OPTIONS", "/completions", headers={
|
|
"Origin": origin,
|
|
"Access-Control-Request-Method": "POST",
|
|
"Access-Control-Request-Headers": "Authorization",
|
|
})
|
|
assert res.status_code == 200
|
|
assert cors_header in res.headers
|
|
assert res.headers[cors_header] == cors_header_value
|
|
|
|
|
|
def test_cors_proxy_only_forwards_explicit_proxy_headers():
|
|
class CaptureHeadersHandler(BaseHTTPRequestHandler):
|
|
def do_GET(self):
|
|
self.server.captured_headers = dict(self.headers)
|
|
self.send_response(200)
|
|
self.end_headers()
|
|
self.wfile.write(b"ok")
|
|
|
|
def log_message(self, format, *args):
|
|
pass
|
|
|
|
target = ThreadingHTTPServer(("127.0.0.1", 0), CaptureHeadersHandler)
|
|
target.captured_headers = {}
|
|
target_thread = threading.Thread(target=target.serve_forever, daemon=True)
|
|
target_thread.start()
|
|
|
|
try:
|
|
server = ServerPreset.tinyllama2()
|
|
server.api_key = TEST_API_KEY
|
|
server.ui_mcp_proxy = True
|
|
server.start()
|
|
|
|
res = server.make_request("GET", f"/cors-proxy?url=http://127.0.0.1:{target.server_port}/capture", headers={
|
|
"Authorization": f"Bearer {TEST_API_KEY}",
|
|
"Proxy-Authorization": "Basic secret",
|
|
"X-Api-Key": TEST_API_KEY,
|
|
"Cookie": "session=secret",
|
|
"x-llama-server-proxy-header-accept": "application/json",
|
|
"x-llama-server-proxy-header-authorization": "Bearer explicit",
|
|
})
|
|
|
|
assert res.status_code == 200
|
|
captured = {key.lower(): value for key, value in target.captured_headers.items()}
|
|
assert captured["accept"] == "application/json"
|
|
assert captured["authorization"] == "Bearer explicit"
|
|
assert "proxy-authorization" not in captured
|
|
assert "x-api-key" not in captured
|
|
assert "cookie" not in captured
|
|
finally:
|
|
target.shutdown()
|
|
target.server_close()
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"media_path, image_url, success",
|
|
[
|
|
(None, "file://mtmd/test-1.jpeg", False), # disabled media path, should fail
|
|
("../../../tools", "file://mtmd/test-1.jpeg", True),
|
|
("../../../tools", "file:////mtmd//test-1.jpeg", True), # should be the same file as above
|
|
("../../../tools", "file://mtmd/notfound.jpeg", False), # non-existent file
|
|
("../../../tools", "file://../mtmd/test-1.jpeg", False), # no directory traversal
|
|
]
|
|
)
|
|
def test_local_media_file(media_path, image_url, success,):
|
|
server = ServerPreset.tinygemma3()
|
|
server.media_path = media_path
|
|
server.start()
|
|
res = server.make_request("POST", "/chat/completions", data={
|
|
"max_tokens": 1,
|
|
"messages": [
|
|
{"role": "user", "content": [
|
|
{"type": "text", "text": "test"},
|
|
{"type": "image_url", "image_url": {
|
|
"url": image_url,
|
|
}},
|
|
]},
|
|
],
|
|
})
|
|
if success:
|
|
assert res.status_code == 200
|
|
else:
|
|
assert res.status_code == 400
|