Compare commits
3 Commits
c10ff428f6
...
4089730bd4
Author | SHA1 | Date | |
---|---|---|---|
4089730bd4 | |||
3270e3fe2c | |||
f209e0e452 |
12
flake.nix
12
flake.nix
@ -19,11 +19,11 @@
|
|||||||
pkgs = forAllSystems (system: inputs.nixpkgs.legacyPackages.${system});
|
pkgs = forAllSystems (system: inputs.nixpkgs.legacyPackages.${system});
|
||||||
poetry2nix = forAllSystems (system: inputs.poetry2nix-lib.lib.mkPoetry2Nix {pkgs = pkgs.${system};});
|
poetry2nix = forAllSystems (system: inputs.poetry2nix-lib.lib.mkPoetry2Nix {pkgs = pkgs.${system};});
|
||||||
|
|
||||||
addSetuptools = self: super: list:
|
addSetuptools = final: prev: list:
|
||||||
builtins.listToAttrs (builtins.map (package: {
|
builtins.listToAttrs (builtins.map (package: {
|
||||||
name = "${package}";
|
name = "${package}";
|
||||||
value = super."${package}".overridePythonAttrs (old: {
|
value = prev."${package}".overridePythonAttrs (old: {
|
||||||
nativeBuildInputs = (old.nativeBuildInputs or []) ++ [self.setuptools];
|
nativeBuildInputs = (old.nativeBuildInputs or []) ++ [final.setuptools];
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
list);
|
list);
|
||||||
@ -32,7 +32,7 @@
|
|||||||
packages = forAllSystems (system: let
|
packages = forAllSystems (system: let
|
||||||
mkPackage = {debug ? false}:
|
mkPackage = {debug ? false}:
|
||||||
poetry2nix.${system}.mkPoetryApplication {
|
poetry2nix.${system}.mkPoetryApplication {
|
||||||
projectDir = self;
|
projectDir = ./.;
|
||||||
checkPhase =
|
checkPhase =
|
||||||
if debug
|
if debug
|
||||||
then "pyright --warnings testdata && pytest"
|
then "pyright --warnings testdata && pytest"
|
||||||
@ -41,8 +41,8 @@
|
|||||||
preferWheels = false;
|
preferWheels = false;
|
||||||
nativeBuildInputs = with pkgs.${system}; [pyright];
|
nativeBuildInputs = with pkgs.${system}; [pyright];
|
||||||
overrides =
|
overrides =
|
||||||
poetry2nix.${system}.overrides.withDefaults (self: super:
|
poetry2nix.${system}.overrides.withDefaults (final: prev:
|
||||||
addSetuptools self super ["sqlite-minutils" "fastlite" "python-fasthtml"]);
|
addSetuptools final prev ["sqlite-minutils" "fastlite" "python-fasthtml"]);
|
||||||
};
|
};
|
||||||
in {
|
in {
|
||||||
default = mkPackage {debug = false;};
|
default = mkPackage {debug = false;};
|
||||||
|
@ -51,21 +51,22 @@ def server():
|
|||||||
|
|
||||||
def test_get_file():
|
def test_get_file():
|
||||||
|
|
||||||
response = requests.get(f'{PROTOCOL}://{HOST}:{PORT}/?api_key={API_KEY}&size=32', timeout = TIMEOUT)
|
response = requests.get(
|
||||||
|
f'{PROTOCOL}://{HOST}:{PORT}/zeros/?api_key={API_KEY}&size=32', timeout=TIMEOUT)
|
||||||
assert response.content == b'\0' * 32
|
assert response.content == b'\0' * 32
|
||||||
|
|
||||||
|
|
||||||
def test_get_file_B():
|
def test_get_file_B():
|
||||||
|
|
||||||
response = requests.get(
|
response = requests.get(
|
||||||
f'{PROTOCOL}://{HOST}:{PORT}/?api_key={API_KEY}&size=32B', timeout=TIMEOUT)
|
f'{PROTOCOL}://{HOST}:{PORT}/zeros/?api_key={API_KEY}&size=32B', timeout=TIMEOUT)
|
||||||
assert response.content == b'\0' * 32
|
assert response.content == b'\0' * 32
|
||||||
|
|
||||||
|
|
||||||
def test_get_file_KB():
|
def test_get_file_KB():
|
||||||
|
|
||||||
response = requests.get(
|
response = requests.get(
|
||||||
f'{PROTOCOL}://{HOST}:{PORT}/?api_key={API_KEY}&size=32KB', timeout=TIMEOUT)
|
f'{PROTOCOL}://{HOST}:{PORT}/zeros/?api_key={API_KEY}&size=32KB', timeout=TIMEOUT)
|
||||||
assert response.status_code == 200
|
assert response.status_code == 200
|
||||||
assert response.content == b'\0' * 32 * 1000
|
assert response.content == b'\0' * 32 * 1000
|
||||||
|
|
||||||
@ -73,7 +74,7 @@ def test_get_file_KB():
|
|||||||
def test_get_file_KiB():
|
def test_get_file_KiB():
|
||||||
|
|
||||||
response = requests.get(
|
response = requests.get(
|
||||||
f'{PROTOCOL}://{HOST}:{PORT}/?api_key={API_KEY}&size=32KiB', timeout=TIMEOUT)
|
f'{PROTOCOL}://{HOST}:{PORT}/zeros/?api_key={API_KEY}&size=32KiB', timeout=TIMEOUT)
|
||||||
assert response.status_code == 200
|
assert response.status_code == 200
|
||||||
assert response.content == b'\0' * 32 * 1024
|
assert response.content == b'\0' * 32 * 1024
|
||||||
|
|
||||||
@ -81,14 +82,15 @@ def test_get_file_KiB():
|
|||||||
def test_get_file_invalid_format():
|
def test_get_file_invalid_format():
|
||||||
|
|
||||||
response = requests.get(
|
response = requests.get(
|
||||||
f'{PROTOCOL}://{HOST}:{PORT}/?api_key={API_KEY}&size=32Invalid', timeout=TIMEOUT)
|
f'{PROTOCOL}://{HOST}:{PORT}/zeros/?api_key={API_KEY}&size=32Invalid', timeout=TIMEOUT)
|
||||||
assert response.status_code == 400
|
assert response.status_code == 400
|
||||||
assert response.text == 'Invalid Format.'
|
assert response.text == 'Invalid Format.'
|
||||||
|
|
||||||
|
|
||||||
def test_database_data_used(server):
|
def test_database_data_used(server):
|
||||||
|
|
||||||
requests.get(f'{PROTOCOL}://{HOST}:{PORT}/?api_key={API_KEY}&size=32', timeout = TIMEOUT)
|
requests.get(
|
||||||
|
f'{PROTOCOL}://{HOST}:{PORT}/zeros/?api_key={API_KEY}&size=32', timeout=TIMEOUT)
|
||||||
with open(server) as file:
|
with open(server) as file:
|
||||||
assert json.loads(file.read())['data-used'] == 32
|
assert json.loads(file.read())['data-used'] == 32
|
||||||
|
|
||||||
|
74
testdata/api.py
vendored
74
testdata/api.py
vendored
@ -1,74 +0,0 @@
|
|||||||
from fasthtml.fastapp import FastHTML
|
|
||||||
from starlette import status
|
|
||||||
from starlette.responses import StreamingResponse
|
|
||||||
from starlette.exceptions import HTTPException
|
|
||||||
from pydantic import ValidationError
|
|
||||||
|
|
||||||
from .utils import load_database, save_database, generate_data
|
|
||||||
from .custom_types import TestDataBody
|
|
||||||
|
|
||||||
def create_api(api_keys: set[str], max_size: int, max_data: int, database: str, buffer_size: int) -> FastHTML:
|
|
||||||
|
|
||||||
api = FastHTML()
|
|
||||||
|
|
||||||
class MaxSizePerRequestError(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class MinSizePerRequestError(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
@api.get('/') # type: ignore
|
|
||||||
async def test_data(api_key: str, size: str) -> StreamingResponse:
|
|
||||||
try:
|
|
||||||
body = TestDataBody(api_key=api_key, size=size) # type: ignore
|
|
||||||
except ValidationError as err:
|
|
||||||
raise HTTPException(
|
|
||||||
status_code=status.HTTP_400_BAD_REQUEST,
|
|
||||||
detail='Invalid Format.'
|
|
||||||
) from err
|
|
||||||
|
|
||||||
try:
|
|
||||||
if body.api_key not in api_keys:
|
|
||||||
raise HTTPException(
|
|
||||||
status_code=status.HTTP_401_UNAUTHORIZED,
|
|
||||||
detail='Invalid API Key.'
|
|
||||||
)
|
|
||||||
|
|
||||||
if body.size < 0:
|
|
||||||
raise MinSizePerRequestError
|
|
||||||
|
|
||||||
if max_size < body.size:
|
|
||||||
raise MaxSizePerRequestError
|
|
||||||
|
|
||||||
db = load_database(database)
|
|
||||||
if max_data <= db['data-used'] + body.size:
|
|
||||||
raise HTTPException(
|
|
||||||
status_code=status.HTTP_500_INTERNAL_SERVER_ERROR,
|
|
||||||
detail='Service not available.'
|
|
||||||
)
|
|
||||||
db['data-used'] += body.size
|
|
||||||
|
|
||||||
save_database(database, db)
|
|
||||||
|
|
||||||
return StreamingResponse(
|
|
||||||
status_code=status.HTTP_200_OK,
|
|
||||||
content=generate_data(body.size, buffer_size),
|
|
||||||
media_type='application/octet-stream',
|
|
||||||
headers={
|
|
||||||
'Content-Length': str(body.size)
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
except MinSizePerRequestError as err:
|
|
||||||
raise HTTPException(
|
|
||||||
status_code=status.HTTP_416_REQUESTED_RANGE_NOT_SATISFIABLE,
|
|
||||||
detail='Size has to be not-negative.'
|
|
||||||
) from err
|
|
||||||
except MaxSizePerRequestError as err:
|
|
||||||
raise HTTPException(
|
|
||||||
status_code=status.HTTP_416_REQUESTED_RANGE_NOT_SATISFIABLE,
|
|
||||||
detail=f'Exceeded max size per request of {max_size} Bytes.'
|
|
||||||
) from err
|
|
||||||
|
|
||||||
return api
|
|
124
testdata/app.py
vendored
Normal file
124
testdata/app.py
vendored
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
# pylint: disable=missing-module-docstring,missing-class-docstring,missing-function-docstring
|
||||||
|
# pylint: disable=line-too-long
|
||||||
|
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from fasthtml.fastapp import FastHTML, FastHTMLWithLiveReload, Request, StaticFiles, picolink
|
||||||
|
from fasthtml.common import Form, Label, Input, Br, Button, A, Html, serve, Body, Link, P, Body, Div, Span, Main, Title # pylint: disable=no-name-in-module
|
||||||
|
from starlette import status
|
||||||
|
from starlette.responses import Response, StreamingResponse
|
||||||
|
from starlette.exceptions import HTTPException
|
||||||
|
from pydantic import ValidationError
|
||||||
|
|
||||||
|
from .utils import load_database, save_database, generate_data
|
||||||
|
from .custom_types import TestDataBody, Config
|
||||||
|
|
||||||
|
|
||||||
|
class MaxSizePerRequestError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class MinSizePerRequestError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
with open('./config.json') as file: # parse_cli_arguments(sys.argv[1:])
|
||||||
|
config = Config.model_validate_json(file.read())
|
||||||
|
|
||||||
|
|
||||||
|
app = FastHTMLWithLiveReload(hdrs=(
|
||||||
|
(Link(rel='stylesheet', type='text/css', href='static/style.css'),)
|
||||||
|
))
|
||||||
|
|
||||||
|
app.mount(
|
||||||
|
'/static',
|
||||||
|
StaticFiles(directory=Path(__file__).parent.absolute() / 'static'),
|
||||||
|
name='static'
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@app.get('/')
|
||||||
|
async def homepage():
|
||||||
|
return Main(
|
||||||
|
P('Testdata Download'),
|
||||||
|
P('Download a file of exact size. The file will only contain zeros.'),
|
||||||
|
Div(
|
||||||
|
Div(
|
||||||
|
Form(
|
||||||
|
Div(
|
||||||
|
Label('API Key'), Br(),
|
||||||
|
Input(name='api_key'), Br()
|
||||||
|
),
|
||||||
|
Div(
|
||||||
|
Label('Size'), Br(),
|
||||||
|
Input(name='size', placeholder='2MiB, 1GB, etc.'), Br()
|
||||||
|
),
|
||||||
|
Button('Download'),
|
||||||
|
action='/zeros', hx_boost="true"
|
||||||
|
),
|
||||||
|
P(f'Max Download Size: {config.max_size / (1024 * 1024 * 1024):.02f} GiB'), Br(), # nopep8
|
||||||
|
cls='download'
|
||||||
|
),
|
||||||
|
P('Contact: ', A('contact.testdata@krsnik.at', href='mailto:contact.testdata@krsnik.at'), id='contact') # nopep8
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ app.get('/zeros') # type: ignore
|
||||||
|
async def get_zeros(api_key: str, size: str, request: Request) -> StreamingResponse:
|
||||||
|
# AJAX Requests cannot return a file, thus send a redirect.
|
||||||
|
if 'HX-Request' in request.headers:
|
||||||
|
return Response(
|
||||||
|
content='<p>Hello</p>',
|
||||||
|
headers={'HX-Redirect': f'/zeros/?api_key={api_key}&size={size}'})
|
||||||
|
|
||||||
|
try:
|
||||||
|
body = TestDataBody(api_key=api_key, size=size) # type: ignore
|
||||||
|
except ValidationError as err:
|
||||||
|
raise HTTPException(
|
||||||
|
status_code=status.HTTP_400_BAD_REQUEST,
|
||||||
|
detail='Invalid Format.'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
try:
|
||||||
|
if body.api_key not in config.api_keys:
|
||||||
|
raise HTTPException(
|
||||||
|
status_code=status.HTTP_401_UNAUTHORIZED,
|
||||||
|
detail='Invalid API Key.'
|
||||||
|
)
|
||||||
|
|
||||||
|
if body.size < 0:
|
||||||
|
raise MinSizePerRequestError
|
||||||
|
|
||||||
|
if config.max_size < body.size:
|
||||||
|
raise MaxSizePerRequestError
|
||||||
|
|
||||||
|
db = load_database(config.database)
|
||||||
|
if config.max_data <= db['data-used'] + body.size:
|
||||||
|
raise HTTPException(
|
||||||
|
status_code=status.HTTP_500_INTERNAL_SERVER_ERROR,
|
||||||
|
detail='Service not available.'
|
||||||
|
)
|
||||||
|
|
||||||
|
db['data-used'] += body.size
|
||||||
|
save_database(config.database, db)
|
||||||
|
|
||||||
|
return StreamingResponse(
|
||||||
|
status_code=status.HTTP_200_OK,
|
||||||
|
content=generate_data(body.size, config.buffer_size),
|
||||||
|
media_type='application/octet-stream',
|
||||||
|
headers={
|
||||||
|
'Content-Length': str(body.size),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
except MinSizePerRequestError as err:
|
||||||
|
raise HTTPException(
|
||||||
|
status_code=status.HTTP_416_REQUESTED_RANGE_NOT_SATISFIABLE,
|
||||||
|
detail='Size has to be not-negative.'
|
||||||
|
) from err
|
||||||
|
except MaxSizePerRequestError as err:
|
||||||
|
raise HTTPException(
|
||||||
|
status_code=status.HTTP_416_REQUESTED_RANGE_NOT_SATISFIABLE,
|
||||||
|
detail=f'Exceeded max size per request of {config.max_size} Bytes.'
|
||||||
|
) from err
|
8
testdata/custom_types.py
vendored
8
testdata/custom_types.py
vendored
@ -20,6 +20,14 @@ class Config(BaseModel):
|
|||||||
def convert_size(cls, value: int | str) -> int:
|
def convert_size(cls, value: int | str) -> int:
|
||||||
return convert_to_bytes(value)
|
return convert_to_bytes(value)
|
||||||
|
|
||||||
|
@field_validator('api_keys', mode='before')
|
||||||
|
@classmethod
|
||||||
|
def check_if_key_in_file(cls, value: str | list[str]) -> set[str]:
|
||||||
|
if isinstance(value, str):
|
||||||
|
with open(value) as file:
|
||||||
|
return set(filter(lambda x: x.strip() != '', file.read().splitlines()))
|
||||||
|
return set(value)
|
||||||
|
|
||||||
|
|
||||||
class TestDataBody(BaseModel):
|
class TestDataBody(BaseModel):
|
||||||
api_key: str
|
api_key: str
|
||||||
|
10
testdata/main.py
vendored
10
testdata/main.py
vendored
@ -1,13 +1,11 @@
|
|||||||
import sys
|
import sys
|
||||||
import argparse
|
import argparse
|
||||||
import json
|
|
||||||
import os
|
import os
|
||||||
from os.path import exists
|
from os.path import exists
|
||||||
|
|
||||||
import uvicorn
|
import uvicorn
|
||||||
|
|
||||||
from .utils import save_database
|
from .utils import save_database
|
||||||
from .api import create_api
|
|
||||||
from .custom_types import Config
|
from .custom_types import Config
|
||||||
|
|
||||||
|
|
||||||
@ -25,16 +23,16 @@ def parse_cli_arguments(argv: list[str]) -> argparse.Namespace:
|
|||||||
return parser.parse_args(argv)
|
return parser.parse_args(argv)
|
||||||
|
|
||||||
|
|
||||||
def run(host: str, port: int, api_keys: set[str], max_size: int, max_data: int, database: str, buffer_size: int):
|
def run(host: str, port: int, api_keys: set[str], max_size: int, max_data: int, database: str, buffer_size: int, reload: bool = False):
|
||||||
if not exists(database) or os.stat(database).st_size == 0:
|
if not exists(database) or os.stat(database).st_size == 0:
|
||||||
save_database(database, {'data-used': 0})
|
save_database(database, {'data-used': 0})
|
||||||
|
|
||||||
api = create_api(api_keys, max_size, max_data, database, buffer_size)
|
# app = create_api(api_keys, max_size, max_data, database, buffer_size)
|
||||||
|
|
||||||
uvicorn.run(api, host = host, port = port)
|
uvicorn.run("testdata.app:app", host=host, port=port, reload=reload)
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
args = parse_cli_arguments(sys.argv[1:])
|
args = parse_cli_arguments(sys.argv[1:])
|
||||||
config = Config.model_validate_json(args.config.read())
|
config = Config.model_validate_json(args.config.read())
|
||||||
run(**config.model_dump(exclude={'log'}))
|
run(**config.model_dump(exclude={'log'}), reload=True)
|
||||||
|
BIN
testdata/static/Roboto-Regular.ttf
vendored
Normal file
BIN
testdata/static/Roboto-Regular.ttf
vendored
Normal file
Binary file not shown.
48
testdata/static/style.css
vendored
Normal file
48
testdata/static/style.css
vendored
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
@font-face {
|
||||||
|
font-family: roboto;
|
||||||
|
src: url(Roboto-Regular.ttf);
|
||||||
|
}
|
||||||
|
|
||||||
|
:root {
|
||||||
|
--background-color: #ffffff;
|
||||||
|
--form-background-color: #e2ddff;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
background-color: var(--background-color);
|
||||||
|
font-family: roboto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.download {
|
||||||
|
background-color: var(--form-background-color);
|
||||||
|
border-radius: 1em;
|
||||||
|
text-align: center;
|
||||||
|
width: 70%;
|
||||||
|
margin: 0 auto;
|
||||||
|
margin-top: 5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.download > form > div {
|
||||||
|
width: 70%;
|
||||||
|
margin: 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.download > form > div > label {
|
||||||
|
font-size: 3em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.download > form > div > input {
|
||||||
|
width: 100%;
|
||||||
|
font-size: 3em;
|
||||||
|
text-indent: 1em;
|
||||||
|
background-color: var(--background-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.download > form > button {
|
||||||
|
margin-top: 1em;
|
||||||
|
font-size: 3em;
|
||||||
|
}
|
||||||
|
|
||||||
|
#contact {
|
||||||
|
text-align: center;
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user