Skip to content

Commit

Permalink
Ensure valid key validation is enabled for all components
Browse files Browse the repository at this point in the history
  • Loading branch information
philippjfr committed Oct 31, 2023
1 parent 592e75b commit f70b7e8
Show file tree
Hide file tree
Showing 11 changed files with 47 additions and 7 deletions.
3 changes: 2 additions & 1 deletion lumen/filters/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
import types

from typing import (
TYPE_CHECKING, Any, Callable, ClassVar, Dict, List, Tuple, Type,
TYPE_CHECKING, Any, Callable, ClassVar, Dict, List, Literal, Tuple, Type,
)

import bokeh # type: ignore
Expand Down Expand Up @@ -60,6 +60,7 @@ class Filter(MultiTypeComponent):
# Specification configuration
_internal_params: ClassVar[List[str]] = ['name', 'schema']
_requires_field: ClassVar[bool] = True
_valid_keys: ClassVar[List[str] | Literal['params'] | None] = 'params'

def __init__(self, **params):
super().__init__(**params)
Expand Down
1 change: 1 addition & 0 deletions lumen/pipeline.py
Original file line number Diff line number Diff line change
Expand Up @@ -145,6 +145,7 @@ class Pipeline(Viewer, Component):

_internal_params: ClassVar[List[str]] = ['data', 'name', 'schema', '_stale']
_required_fields: ClassVar[List[str | Tuple[str, str]]] = [('source', 'pipeline')]
_valid_keys: ClassVar[List[str] | Literal['params'] | None] = 'params'

def __init__(self, *, source, table, **params):
if 'schema' not in params:
Expand Down
5 changes: 4 additions & 1 deletion lumen/sources/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@
from os.path import basename
from pathlib import Path
from typing import (
TYPE_CHECKING, Any, ClassVar, Dict, List, Tuple, Type, Union,
TYPE_CHECKING, Any, ClassVar, Dict, List, Literal, Tuple, Type, Union,
)
from urllib.parse import quote, urlparse

Expand Down Expand Up @@ -170,6 +170,9 @@ class Source(MultiTypeComponent):
# Declare whether source supports SQL transforms
_supports_sql: ClassVar[bool] = False

# Valid keys incude all parameters (except _internal_params)
_valid_keys: ClassVar[List[str] | Literal['params'] | None] = 'params'

@property
def _reload_params(self) -> List[str]:
"List of parameters that trigger a data reload."
Expand Down
7 changes: 6 additions & 1 deletion lumen/tests/validation/test_filters.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@
"The ConstantFilter component requires 'field' parameter to be defined",
),
(
{"type": "widget", "fields": "island"},
{"type": "widget"},
"The WidgetFilter component requires 'field' parameter to be defined",
),
(
Expand All @@ -41,3 +41,8 @@ def test_filter_Filter(spec, msg):
else:
with pytest.raises(ValidationError, match=msg):
Filter.validate(spec)


def test_filter_key_validation():
with pytest.raises(ValidationError, match="ConstantFilter component specification contained unknown key 'feld'"):
Filter.validate({'type': 'constant', 'feld': 'foo'})
5 changes: 5 additions & 0 deletions lumen/tests/validation/test_layout.py
Original file line number Diff line number Diff line change
Expand Up @@ -104,3 +104,8 @@ def test_layout_Layout(spec, msg):
else:
with pytest.raises(ValidationError, match=msg):
Layout.validate(spec, context)


def test_layout_key_validation():
with pytest.raises(ValidationError, match="Layout component specification contained unknown key 'src'"):
Layout.validate({'title': 'Table', 'src': 'penguins'})
5 changes: 5 additions & 0 deletions lumen/tests/validation/test_pipeline.py
Original file line number Diff line number Diff line change
Expand Up @@ -100,3 +100,8 @@ def test_pipeline_Pipeline(source, filters, transforms, msg):
else:
with pytest.raises(ValidationError, match=msg):
Pipeline.validate(spec)


def test_pipeline_key_validation():
with pytest.raises(ValidationError, match="Pipeline component specification contained unknown key 'transfomers'"):
Pipeline.validate({'source': None, 'transfomers': []})
5 changes: 5 additions & 0 deletions lumen/tests/validation/test_sources.py
Original file line number Diff line number Diff line change
Expand Up @@ -57,3 +57,8 @@ def test_source_Source(spec, context, msg):
else:
with pytest.raises(ValidationError, match=msg):
Source.validate(spec, context)


def test_source_key_validation():
with pytest.raises(ValidationError, match="FileSource component specification contained unknown key 'tablas'"):
Source.validate({'type': 'file', 'tablas': {}})
4 changes: 4 additions & 0 deletions lumen/tests/validation/test_transform.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,3 +29,7 @@ def test_transforms_Transform(spec, msg):
else:
with pytest.raises(ValidationError, match=msg):
Transform.validate(spec)

def test_transform_key_validation():
with pytest.raises(ValidationError, match="Iloc component specification contained unknown key 'strt'"):
Transform.validate({'type': 'iloc', 'strt': 3})
9 changes: 7 additions & 2 deletions lumen/tests/validation/test_views.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@
{"format": "csv", "type": "default"},
),
(
{"formats": "csv"},
{},
"The Download component requires 'format' parameter to be defined",
),
(
Expand Down Expand Up @@ -46,7 +46,7 @@ def test_target_Download(spec, output):
None,
),
(
{"type": "download", "formats": "csv"},
{"type": "download"},
"The DownloadView component requires 'format' parameter to be defined",
),
(
Expand All @@ -71,3 +71,8 @@ def test_target_View(spec, msg):
else:
with pytest.raises(ValidationError, match=msg):
View.validate(spec)


def test_filter_key_validation():
with pytest.raises(ValidationError, match="Table component specification contained unknown key 'feld'"):
View.validate({'type': 'table', 'feld': 'foo'})
4 changes: 3 additions & 1 deletion lumen/transforms/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
import hashlib

from typing import (
TYPE_CHECKING, Any, Callable, ClassVar, Dict, List, Tuple, Union,
TYPE_CHECKING, Any, Callable, ClassVar, Dict, List, Literal, Tuple, Union,
)

import numpy as np
Expand Down Expand Up @@ -43,6 +43,8 @@ class Transform(MultiTypeComponent):

_field_params: ClassVar[List[str]] = []

_valid_keys: ClassVar[List[str] | Literal['params'] | None] = 'params'

__abstract = True

@classmethod
Expand Down
6 changes: 5 additions & 1 deletion lumen/views/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@

from io import BytesIO, StringIO
from typing import (
IO, TYPE_CHECKING, Any, ClassVar, Dict, List, Type,
IO, TYPE_CHECKING, Any, ClassVar, Dict, List, Literal, Type,
)
from weakref import WeakKeyDictionary

Expand Down Expand Up @@ -93,6 +93,8 @@ class View(MultiTypeComponent, Viewer):
# Panel extension to load to render this View
_extension: ClassVar[str | None] = None

_internal_params: ClassVar[List[str]] = ['rerender']

# Parameters which reference fields in the table
_field_params: ClassVar[List[str]] = ['field']

Expand All @@ -104,6 +106,8 @@ class View(MultiTypeComponent, Viewer):

_panel_type: ClassVar[Type[Viewable] | None] = None

_valid_keys: ClassVar[List[str] | Literal['params'] | None] = 'params'

__abstract = True

def __init__(self, **params):
Expand Down

0 comments on commit f70b7e8

Please sign in to comment.