mirror of
https://github.com/mofeng-git/One-KVM.git
synced 2025-12-12 01:00:29 +08:00
moar validators
This commit is contained in:
parent
4eb89c9399
commit
07c59485fc
@ -51,9 +51,12 @@ from ..validators.basic import valid_number
|
|||||||
from ..validators.basic import valid_int_f1
|
from ..validators.basic import valid_int_f1
|
||||||
from ..validators.basic import valid_float_f01
|
from ..validators.basic import valid_float_f01
|
||||||
|
|
||||||
|
from ..validators.auth import valid_users_list
|
||||||
|
|
||||||
from ..validators.fs import valid_abs_path
|
from ..validators.fs import valid_abs_path
|
||||||
from ..validators.fs import valid_abs_path_exists
|
from ..validators.fs import valid_abs_path_exists
|
||||||
from ..validators.fs import valid_unix_mode
|
from ..validators.fs import valid_unix_mode
|
||||||
|
from ..validators.fs import valid_command
|
||||||
|
|
||||||
from ..validators.net import valid_ip_or_host
|
from ..validators.net import valid_ip_or_host
|
||||||
from ..validators.net import valid_port
|
from ..validators.net import valid_port
|
||||||
@ -150,7 +153,7 @@ def _get_config_scheme() -> Dict:
|
|||||||
},
|
},
|
||||||
|
|
||||||
"auth": {
|
"auth": {
|
||||||
"internal_users": Option([]),
|
"internal_users": Option([], type=valid_users_list),
|
||||||
"internal_type": Option("htpasswd"),
|
"internal_type": Option("htpasswd"),
|
||||||
"external_type": Option(""),
|
"external_type": Option(""),
|
||||||
# "internal": {},
|
# "internal": {},
|
||||||
@ -222,7 +225,7 @@ def _get_config_scheme() -> Dict:
|
|||||||
"unix": Option("", type=valid_abs_path, only_if="!port", unpack_as="unix_path"),
|
"unix": Option("", type=valid_abs_path, only_if="!port", unpack_as="unix_path"),
|
||||||
"timeout": Option(2.0, type=valid_float_f01),
|
"timeout": Option(2.0, type=valid_float_f01),
|
||||||
|
|
||||||
"cmd": Option(["/bin/true"]), # TODO: Validator
|
"cmd": Option(["/bin/true"], type=valid_command),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@ -20,8 +20,11 @@
|
|||||||
# ========================================================================== #
|
# ========================================================================== #
|
||||||
|
|
||||||
|
|
||||||
|
from typing import List
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
|
from .basic import valid_string_list
|
||||||
|
|
||||||
from . import check_re_match
|
from . import check_re_match
|
||||||
|
|
||||||
|
|
||||||
@ -30,6 +33,10 @@ def valid_user(arg: Any) -> str:
|
|||||||
return check_re_match(arg, "username characters", r"^[a-z_][a-z0-9_-]*$")
|
return check_re_match(arg, "username characters", r"^[a-z_][a-z0-9_-]*$")
|
||||||
|
|
||||||
|
|
||||||
|
def valid_users_list(arg: Any) -> List[str]:
|
||||||
|
return valid_string_list(arg, subval=valid_user, name="users list")
|
||||||
|
|
||||||
|
|
||||||
def valid_passwd(arg: Any) -> str:
|
def valid_passwd(arg: Any) -> str:
|
||||||
return check_re_match(arg, "passwd characters", r"^[\x20-\x7e]*\Z$", strip=False, hide=True)
|
return check_re_match(arg, "passwd characters", r"^[\x20-\x7e]*\Z$", strip=False, hide=True)
|
||||||
|
|
||||||
|
|||||||
@ -20,7 +20,12 @@
|
|||||||
# ========================================================================== #
|
# ========================================================================== #
|
||||||
|
|
||||||
|
|
||||||
|
import re
|
||||||
|
|
||||||
|
from typing import List
|
||||||
from typing import Type
|
from typing import Type
|
||||||
|
from typing import Callable
|
||||||
|
from typing import Optional
|
||||||
from typing import Union
|
from typing import Union
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
@ -71,3 +76,27 @@ def valid_int_f1(arg: Any) -> int:
|
|||||||
|
|
||||||
def valid_float_f01(arg: Any) -> float:
|
def valid_float_f01(arg: Any) -> float:
|
||||||
return float(valid_number(arg, min=0.1, type=float))
|
return float(valid_number(arg, min=0.1, type=float))
|
||||||
|
|
||||||
|
|
||||||
|
def valid_string_list(
|
||||||
|
arg: Any,
|
||||||
|
delim: str=r"[,\t ]+",
|
||||||
|
subval: Optional[Callable[[Any], Any]]=None,
|
||||||
|
name: str="",
|
||||||
|
) -> List[str]:
|
||||||
|
|
||||||
|
if not name:
|
||||||
|
name = "string list"
|
||||||
|
|
||||||
|
if subval is None:
|
||||||
|
subval = (lambda item: check_not_none_string(item, name + " item"))
|
||||||
|
|
||||||
|
if not isinstance(arg, (list, tuple)):
|
||||||
|
arg = check_not_none_string(arg, name)
|
||||||
|
arg = list(filter(None, re.split(delim, arg)))
|
||||||
|
if subval is not None:
|
||||||
|
try:
|
||||||
|
arg = list(map(subval, arg))
|
||||||
|
except Exception:
|
||||||
|
raise ValidatorError("Failed sub-validator on one of the item of %r" % (arg))
|
||||||
|
return arg
|
||||||
|
|||||||
@ -22,16 +22,19 @@
|
|||||||
|
|
||||||
import os
|
import os
|
||||||
|
|
||||||
|
from typing import List
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from . import raise_error
|
from . import raise_error
|
||||||
from . import check_not_none_string
|
from . import check_not_none_string
|
||||||
|
|
||||||
from .basic import valid_number
|
from .basic import valid_number
|
||||||
|
from .basic import valid_string_list
|
||||||
|
|
||||||
|
|
||||||
# =====
|
# =====
|
||||||
def valid_abs_path(arg: Any, exists: bool=False) -> str:
|
def valid_abs_path(arg: Any, exists: bool=False, name: str="") -> str:
|
||||||
|
if not name:
|
||||||
name = ("existent absolute path" if exists else "absolute path")
|
name = ("existent absolute path" if exists else "absolute path")
|
||||||
|
|
||||||
if len(str(arg).strip()) == 0:
|
if len(str(arg).strip()) == 0:
|
||||||
@ -44,9 +47,17 @@ def valid_abs_path(arg: Any, exists: bool=False) -> str:
|
|||||||
return arg
|
return arg
|
||||||
|
|
||||||
|
|
||||||
def valid_abs_path_exists(arg: Any) -> str:
|
def valid_abs_path_exists(arg: Any, name: str="") -> str:
|
||||||
return valid_abs_path(arg, exists=True)
|
return valid_abs_path(arg, exists=True, name=name)
|
||||||
|
|
||||||
|
|
||||||
def valid_unix_mode(arg: Any) -> int:
|
def valid_unix_mode(arg: Any) -> int:
|
||||||
return int(valid_number(arg, min=0, name="UNIX mode"))
|
return int(valid_number(arg, min=0, name="UNIX mode"))
|
||||||
|
|
||||||
|
|
||||||
|
def valid_command(arg: Any) -> List[str]:
|
||||||
|
cmd = valid_string_list(arg, delim=r"[,\t]+", name="command")
|
||||||
|
if len(cmd) == 0:
|
||||||
|
raise_error(arg, "command")
|
||||||
|
cmd[0] = valid_abs_path_exists(cmd[0], name="command entry point")
|
||||||
|
return cmd
|
||||||
|
|||||||
@ -35,7 +35,8 @@ from kvmd.apps.cleanup import main
|
|||||||
def test_ok(tmpdir) -> None: # type: ignore
|
def test_ok(tmpdir) -> None: # type: ignore
|
||||||
queue: multiprocessing.queues.Queue = multiprocessing.Queue()
|
queue: multiprocessing.queues.Queue = multiprocessing.Queue()
|
||||||
|
|
||||||
ustreamer_fake_name = "ustr-" + secrets.token_hex(3)
|
ustreamer_tmp_path = os.path.abspath(str(tmpdir.join("ustr-" + secrets.token_hex(3))))
|
||||||
|
os.symlink("/usr/bin/ustreamer", ustreamer_tmp_path)
|
||||||
|
|
||||||
ustreamer_sock_path = os.path.abspath(str(tmpdir.join("ustreamer-fake.sock")))
|
ustreamer_sock_path = os.path.abspath(str(tmpdir.join("ustreamer-fake.sock")))
|
||||||
open(ustreamer_sock_path, "w").close()
|
open(ustreamer_sock_path, "w").close()
|
||||||
@ -43,7 +44,7 @@ def test_ok(tmpdir) -> None: # type: ignore
|
|||||||
open(kvmd_sock_path, "w").close()
|
open(kvmd_sock_path, "w").close()
|
||||||
|
|
||||||
def ustreamer_fake() -> None:
|
def ustreamer_fake() -> None:
|
||||||
setproctitle.setproctitle(ustreamer_fake_name)
|
setproctitle.setproctitle(os.path.basename(ustreamer_tmp_path))
|
||||||
queue.put(True)
|
queue.put(True)
|
||||||
while True:
|
while True:
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
@ -60,7 +61,7 @@ def test_ok(tmpdir) -> None: # type: ignore
|
|||||||
"kvmd/server/unix=" + kvmd_sock_path,
|
"kvmd/server/unix=" + kvmd_sock_path,
|
||||||
"kvmd/streamer/port=0",
|
"kvmd/streamer/port=0",
|
||||||
"kvmd/streamer/unix=" + ustreamer_sock_path,
|
"kvmd/streamer/unix=" + ustreamer_sock_path,
|
||||||
"kvmd/streamer/cmd=[\"%s\"]" % (ustreamer_fake_name),
|
"kvmd/streamer/cmd=" + ustreamer_tmp_path,
|
||||||
])
|
])
|
||||||
|
|
||||||
assert not os.path.exists(ustreamer_sock_path)
|
assert not os.path.exists(ustreamer_sock_path)
|
||||||
|
|||||||
@ -20,12 +20,14 @@
|
|||||||
# ========================================================================== #
|
# ========================================================================== #
|
||||||
|
|
||||||
|
|
||||||
|
from typing import List
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from kvmd.validators import ValidatorError
|
from kvmd.validators import ValidatorError
|
||||||
from kvmd.validators.auth import valid_user
|
from kvmd.validators.auth import valid_user
|
||||||
|
from kvmd.validators.auth import valid_users_list
|
||||||
from kvmd.validators.auth import valid_passwd
|
from kvmd.validators.auth import valid_passwd
|
||||||
from kvmd.validators.auth import valid_auth_token
|
from kvmd.validators.auth import valid_auth_token
|
||||||
|
|
||||||
@ -58,6 +60,27 @@ def test_fail__valid_user(arg: Any) -> None:
|
|||||||
print(valid_user(arg))
|
print(valid_user(arg))
|
||||||
|
|
||||||
|
|
||||||
|
# =====
|
||||||
|
@pytest.mark.parametrize("arg, retval", [
|
||||||
|
("foo, bar, ", ["foo", "bar"]),
|
||||||
|
("foo bar", ["foo", "bar"]),
|
||||||
|
(["foo", "bar"], ["foo", "bar"]),
|
||||||
|
("", []),
|
||||||
|
(" ", []),
|
||||||
|
(", ", []),
|
||||||
|
(", foo, ", ["foo"]),
|
||||||
|
([], []),
|
||||||
|
])
|
||||||
|
def test_ok__valid_users_list(arg: Any, retval: List) -> None:
|
||||||
|
assert valid_users_list(arg) == retval
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("arg", [None, [None], [""], [" "], ["user,"]])
|
||||||
|
def test_fail__valid_users_list(arg: Any) -> None: # pylint: disable=invalid-name
|
||||||
|
with pytest.raises(ValidatorError):
|
||||||
|
print(valid_users_list(arg))
|
||||||
|
|
||||||
|
|
||||||
# =====
|
# =====
|
||||||
@pytest.mark.parametrize("arg", [
|
@pytest.mark.parametrize("arg", [
|
||||||
"glados",
|
"glados",
|
||||||
|
|||||||
@ -20,6 +20,7 @@
|
|||||||
# ========================================================================== #
|
# ========================================================================== #
|
||||||
|
|
||||||
|
|
||||||
|
from typing import List
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
@ -29,6 +30,7 @@ from kvmd.validators.basic import valid_bool
|
|||||||
from kvmd.validators.basic import valid_number
|
from kvmd.validators.basic import valid_number
|
||||||
from kvmd.validators.basic import valid_int_f1
|
from kvmd.validators.basic import valid_int_f1
|
||||||
from kvmd.validators.basic import valid_float_f01
|
from kvmd.validators.basic import valid_float_f01
|
||||||
|
from kvmd.validators.basic import valid_string_list
|
||||||
|
|
||||||
|
|
||||||
# =====
|
# =====
|
||||||
@ -105,3 +107,38 @@ def test_ok__valid_float_f01(arg: Any) -> None:
|
|||||||
def test_fail__valid_float_f01(arg: Any) -> None:
|
def test_fail__valid_float_f01(arg: Any) -> None:
|
||||||
with pytest.raises(ValidatorError):
|
with pytest.raises(ValidatorError):
|
||||||
print(valid_float_f01(arg))
|
print(valid_float_f01(arg))
|
||||||
|
|
||||||
|
|
||||||
|
# =====
|
||||||
|
@pytest.mark.parametrize("arg, retval", [
|
||||||
|
("a, b, c", ["a", "b", "c"]),
|
||||||
|
("a b c", ["a", "b", "c"]),
|
||||||
|
(["a", "b", "c"], ["a", "b", "c"]),
|
||||||
|
("", []),
|
||||||
|
(" ", []),
|
||||||
|
(", ", []),
|
||||||
|
(", a, ", ["a"]),
|
||||||
|
([], []),
|
||||||
|
])
|
||||||
|
def test_ok__valid_string_list(arg: Any, retval: List) -> None:
|
||||||
|
assert valid_string_list(arg) == retval
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("arg, retval", [
|
||||||
|
("1, 2, 3", [1, 2, 3]),
|
||||||
|
("1 2 3", [1, 2, 3]),
|
||||||
|
([1, 2, 3], [1, 2, 3]),
|
||||||
|
("", []),
|
||||||
|
(" ", []),
|
||||||
|
(", ", []),
|
||||||
|
(", 1, ", [1]),
|
||||||
|
([], []),
|
||||||
|
])
|
||||||
|
def test_ok__valid_string_list__subval(arg: Any, retval: List) -> None: # pylint: disable=invalid-name
|
||||||
|
assert valid_string_list(arg, subval=int) == retval
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("arg", [None, [None]])
|
||||||
|
def test_fail__valid_string_list(arg: Any) -> None: # pylint: disable=invalid-name
|
||||||
|
with pytest.raises(ValidatorError):
|
||||||
|
print(valid_string_list(arg))
|
||||||
|
|||||||
@ -22,6 +22,7 @@
|
|||||||
|
|
||||||
import os
|
import os
|
||||||
|
|
||||||
|
from typing import List
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
@ -30,6 +31,7 @@ from kvmd.validators import ValidatorError
|
|||||||
from kvmd.validators.fs import valid_abs_path
|
from kvmd.validators.fs import valid_abs_path
|
||||||
from kvmd.validators.fs import valid_abs_path_exists
|
from kvmd.validators.fs import valid_abs_path_exists
|
||||||
from kvmd.validators.fs import valid_unix_mode
|
from kvmd.validators.fs import valid_unix_mode
|
||||||
|
from kvmd.validators.fs import valid_command
|
||||||
|
|
||||||
|
|
||||||
# =====
|
# =====
|
||||||
@ -89,3 +91,28 @@ def test_ok__valid_unix_mode(arg: Any) -> None:
|
|||||||
def test_fail__valid_unix_mode(arg: Any) -> None:
|
def test_fail__valid_unix_mode(arg: Any) -> None:
|
||||||
with pytest.raises(ValidatorError):
|
with pytest.raises(ValidatorError):
|
||||||
print(valid_unix_mode(arg))
|
print(valid_unix_mode(arg))
|
||||||
|
|
||||||
|
|
||||||
|
# =====
|
||||||
|
@pytest.mark.parametrize("arg, retval", [
|
||||||
|
(["/bin/true"], ["/bin/true"]),
|
||||||
|
(["/bin/true", 1, 2, 3], ["/bin/true", "1", "2", "3"]),
|
||||||
|
("/bin/true, 1, 2, 3,", ["/bin/true", "1", "2", "3"]),
|
||||||
|
("/bin/true", ["/bin/true"]),
|
||||||
|
])
|
||||||
|
def test_ok__valid_command(arg: Any, retval: List[str]) -> None:
|
||||||
|
assert valid_command(arg) == retval
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("arg", [
|
||||||
|
["/bin/blahblahblah"],
|
||||||
|
["/bin/blahblahblah", 1, 2, 3],
|
||||||
|
[" "],
|
||||||
|
[],
|
||||||
|
"/bin/blahblahblah, 1, 2, 3,",
|
||||||
|
"/bin/blahblahblah",
|
||||||
|
" ",
|
||||||
|
])
|
||||||
|
def test_fail__valid_command(arg: Any) -> None:
|
||||||
|
with pytest.raises(ValidatorError):
|
||||||
|
print(valid_command(arg))
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user