Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor: Add ruff rules, improve type annotations, improve ci performance #3431

Merged
merged 101 commits into from
Jun 27, 2024
Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
101 commits
Select commit Hold shift + click to select a range
de50aa9
ci: Add additional `ruff` rules to `pyproject.toml`
dangotbanned Jun 5, 2024
e019f17
fix: add item to `pyproject.toml` to silence import errors for `pylan…
dangotbanned Jun 5, 2024
54815ce
ci: add subset of `SIM` to `extend-safe-fixes`
dangotbanned Jun 5, 2024
1effcb1
ci: add unfixable `RUF` rules to `ignore`
dangotbanned Jun 5, 2024
a3fd77a
refactor: apply new `ruff` rules, fix and reformat
dangotbanned Jun 5, 2024
b6190f2
test: Skip tests on Win that require a tz database
dangotbanned Jun 5, 2024
0c57781
ci: enable `tool.ruff.lint.preview`
dangotbanned Jun 5, 2024
3171a3d
fix: replace [F841](https://docs.astral.sh/ruff/rules/unused-variable…
dangotbanned Jun 5, 2024
5ba210d
ci: add additional `preview` fixes to `extend-safe-fixes`
dangotbanned Jun 5, 2024
8ef6184
refactor: apply `preview` fixes for existing `ruff` rules, reformat
dangotbanned Jun 5, 2024
a16b292
ci: add `preview` category `FURB` rules
dangotbanned Jun 5, 2024
7fff541
refactor: apply `FURB` rule fixes, manually fix `FURB101/3`
dangotbanned Jun 5, 2024
217208b
fix: Revert newer fstring syntax, not available to `sphinx`
dangotbanned Jun 5, 2024
1e17533
ci: add fixable `pylint` rules
dangotbanned Jun 5, 2024
007b8b3
ci: add `pylint` fixes to `extend-safe-fixes`
dangotbanned Jun 5, 2024
10eb5d5
refactor: apply `pylint` rule fixes, add an inline optimization for `…
dangotbanned Jun 5, 2024
e1ab766
fix: Recover comments lost during linting
dangotbanned Jun 5, 2024
f3f6196
fix: Replace sources of `RUF002` violations
dangotbanned Jun 5, 2024
d4a554b
fix: manual fix `RUF002` in `api`
dangotbanned Jun 5, 2024
d584e08
ci: add `PTH` rules
dangotbanned Jun 6, 2024
ed2528b
refactor: Manually fix `PTH` rule violations
dangotbanned Jun 6, 2024
8393a5a
fix: Use safer `inspect.getattr_static` in `update_init_file`
dangotbanned Jun 6, 2024
38a3280
fix: Use correct f-string `!s` modifier
dangotbanned Jun 6, 2024
c7485dc
fix: Resolve `doc:build-html` error
dangotbanned Jun 6, 2024
626bc20
fix: Resolve utf-8 error for [emoji example](https://altair-viz.githu…
dangotbanned Jun 6, 2024
8e6a523
Update sphinxext/altairgallery.py
dangotbanned Jun 6, 2024
e935ad9
build: bump `docbuild.yml` python `3.10` -> `3.12`
dangotbanned Jun 7, 2024
9073a1f
style: Simplify `PluginRegistry` repr
dangotbanned Jun 7, 2024
de01aa5
revert: ignore `suppressible-exception` rule in `pyproject.toml`
dangotbanned Jun 7, 2024
80a7a22
refactor: remove need for exception handling in `utils._vegafusion_da…
dangotbanned Jun 7, 2024
b96c4e9
refactor: remove `python<3.3` compat code in `utils.core.use_signature`
dangotbanned Jun 7, 2024
88ec5c8
revert: replace `PLW1514` fixes with "utf-8"
dangotbanned Jun 7, 2024
59893bd
refactor: minor simplify and sort `__all__` from `generate_schema_wra…
dangotbanned Jun 7, 2024
03b4d09
docs: `FA100` on most of `tools/`*`
dangotbanned Jun 7, 2024
e17382a
docs: `FA100` on `sphinxext/*`
dangotbanned Jun 7, 2024
6f02e2e
docs: `FA100` on `alt.jupyter`
dangotbanned Jun 7, 2024
12a9bea
docs: `FA100` on `alt.expr`
dangotbanned Jun 7, 2024
c378b65
docs: `FA100` on `alt.vegalite`, excluding `v5.api`, `v5.schema/*`
dangotbanned Jun 8, 2024
f6da84e
docs: `FA100` on private `alt.utils` modules
dangotbanned Jun 8, 2024
185f2fd
docs(typing): Add annotations for `sphinxext`
dangotbanned Jun 8, 2024
e52e414
docs: `FA100` on public `alt.utils` modules, excluding `schemapi`
dangotbanned Jun 8, 2024
811d566
ci: update `pyproject.toml`, add new granular `tool.ruff.lint.per-fil…
dangotbanned Jun 8, 2024
dba02c9
revert: Change `write_file_or_filename` default `encoding` to `None`
dangotbanned Jun 8, 2024
a8597cb
docs: `FA100` on `tools.schemapi.schemapi`
dangotbanned Jun 8, 2024
6cfc769
feat(typing): add `from __future__ import annotations` for each file …
dangotbanned Jun 8, 2024
2a06b2e
ci(typing): adds `generate-schema-wrapper` script to `hatch`
dangotbanned Jun 9, 2024
337d844
refactor(typing): Improve annotations and narrowing in `schemapi`
dangotbanned Jun 9, 2024
76bc00b
build: run `generate-schema-wrapper`
dangotbanned Jun 9, 2024
ece2e85
revert(typing): Roll back runtime evaluated types not possible on old…
dangotbanned Jun 10, 2024
62f5eeb
fix(typing): Adds overloads and reduce ignore comments relating to `s…
dangotbanned Jun 10, 2024
eb5b30d
revert(typing): Roll back additional runtime evaluated types for `pyt…
dangotbanned Jun 10, 2024
d8a36e6
fix(typing): Use `...` for `DataTransformerType`
dangotbanned Jun 10, 2024
ea5a3d4
docs(typing): `FA100` on `alt.vegalite.v5.api` and manual annotation …
dangotbanned Jun 10, 2024
70e8e95
test: add pytest rules `PT` and fix `PT001` violations
dangotbanned Jun 10, 2024
8bbb78e
test: add ignores for `PT011` on existing tests
dangotbanned Jun 10, 2024
cd56353
test: fix `PT018` violation
dangotbanned Jun 10, 2024
27aab26
test: fix `PT006` violations
dangotbanned Jun 10, 2024
a820ed1
test: add ignore for `PT012` violation
dangotbanned Jun 10, 2024
4b38b87
test: add `pytest.mark.xfail` for flaky `scatter_with_layered_histogr…
dangotbanned Jun 10, 2024
25ce09b
ci: tidy up `ruff` section of `pyproject.toml`
dangotbanned Jun 10, 2024
fd20f00
perf: adds config `pyproject.toml` for faster build, test runs
dangotbanned Jun 10, 2024
ad9e70c
ci: adds `update-init-file` hatch script
dangotbanned Jun 10, 2024
8308b6e
ci: adds newer-style `hatch` test config
dangotbanned Jun 10, 2024
36da1c7
feat(typing): Use `TYPE_CHECKING` block for `schema` modules
dangotbanned Jun 10, 2024
538b527
test: use a more reliable `xfail` condition for flaky test
dangotbanned Jun 11, 2024
9582971
build: Embed extra `ruff` calls in `generate-schema-wrapper` into the…
dangotbanned Jun 11, 2024
33056a2
build: Manually rewrite certain exceptions with flaky autofix
dangotbanned Jun 11, 2024
5244100
refactor: remove now-unneeded `PARAMETER_PROTOCOL`
dangotbanned Jun 11, 2024
4dba0a6
refactor: replace existing references to `typing.Optional`
dangotbanned Jun 11, 2024
03c217c
refactor(typing): rename `T` TypeVar to `TSchemaBase`
dangotbanned Jun 11, 2024
fbc02e2
feat(typing): Adds dedicated `Optional` alias for `Union[..., Undefin…
dangotbanned Jun 11, 2024
f8aa86f
refactor(typing): Remove `UndefinedType` dependency in `api`
dangotbanned Jun 12, 2024
046faa8
refactor(typing): Remove `UndefinedType` dependency in `api`
dangotbanned Jun 12, 2024
05ae13f
refactor(typing): Remove annotation scope `UndefinedType` dependency …
dangotbanned Jun 12, 2024
30ff371
ci: add `W291` to ensure trailing whitespace is autofixed
dangotbanned Jun 12, 2024
dc98ae1
refactor: define non-relevant attributes closer to imports in `update…
dangotbanned Jun 12, 2024
f82adbd
feat(typing): Adds `_TypeAliasTracer` and reorders `generate_schema_w…
dangotbanned Jun 12, 2024
a7b0ab5
build: run `generate-schema-wrapper` using `_TypeAliasTracer`
dangotbanned Jun 12, 2024
4593796
fix: Add missing `LiteralString` import
dangotbanned Jun 12, 2024
3e07902
test: add `pytest.mark.filterwarnings` for tests that cannot avoid them
dangotbanned Jun 13, 2024
7cf20c0
refactor(typing): Replace `Literal[None]` -> `None`
dangotbanned Jun 13, 2024
65e3279
test: Change `skipif` -> `xfail` for `test_sanitize_pyarrow_table_col…
dangotbanned Jun 13, 2024
60b40a5
ci: bump `actions/setup-python`, `python-version`, use `uv` in `lint.…
dangotbanned Jun 14, 2024
ce7c95b
ci: create venv before uv pip install
dangotbanned Jun 14, 2024
50cd1e3
ci: ensure venv is activated after install
dangotbanned Jun 14, 2024
330f638
ci: use environment provided by `hatch`
dangotbanned Jun 14, 2024
237079f
ci: testing `pytest-xdist` in `build` workflow
dangotbanned Jun 14, 2024
3ec3e6a
test(perf): adds significant parallelism for slow `test_examples`
dangotbanned Jun 14, 2024
0da50cc
refactor, perf: reduce `sys.path` usage, use dictcomp in `update_init…
dangotbanned Jun 15, 2024
686a84b
build: adding debug message to help with build failure
dangotbanned Jun 15, 2024
5da0207
build: add cwd to debug message
dangotbanned Jun 15, 2024
eede0f4
fix: possibly fix cwd not on path
dangotbanned Jun 15, 2024
89ab1f3
ci: testing invoking script with `-m` for cwd
dangotbanned Jun 15, 2024
87a686c
fix: remove debug code
dangotbanned Jun 15, 2024
2122dd2
Merge branch 'update-ruff-rules' of https://github.com/dangotbanned/a…
dangotbanned Jun 15, 2024
580c9f6
Merge branch 'main' into update-ruff-rules
dangotbanned Jun 19, 2024
d0c01eb
fix: resolve lint, format, type conflicts following rebase
dangotbanned Jun 19, 2024
ca961c0
DO NOT MERGE - TESTING DOC PERF
dangotbanned Jun 21, 2024
26df622
revert: undo last commit
dangotbanned Jun 21, 2024
bb32cd3
refactor: Remove commented out dead code
dangotbanned Jun 27, 2024
89e1029
ci: Remove extra whitespace in `build.yml`
dangotbanned Jun 27, 2024
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
15 changes: 8 additions & 7 deletions altair/_magics.py
Original file line number Diff line number Diff line change
Expand Up @@ -45,22 +45,22 @@ def _prepare_data(data, data_transformers):
elif isinstance(data, str):
return {"url": data}
else:
warnings.warn("data of type {} not recognized".format(type(data)), stacklevel=1)
warnings.warn(f"data of type {type(data)} not recognized", stacklevel=1)
return data


def _get_variable(name):
"""Get a variable from the notebook namespace."""
ip = IPython.get_ipython()
if ip is None:
raise ValueError(
msg = (
"Magic command must be run within an IPython "
"environment, in which get_ipython() is defined."
)
raise ValueError(msg)
if name not in ip.user_ns:
raise NameError(
"argument '{}' does not match the name of any defined variable".format(name)
)
msg = f"argument '{name}' does not match the name of any defined variable"
raise NameError(msg)
return ip.user_ns[name]


Expand Down Expand Up @@ -95,10 +95,11 @@ def vegalite(line, cell):
try:
spec = json.loads(cell)
except json.JSONDecodeError as err:
raise ValueError(
msg = (
"%%vegalite: spec is not valid JSON. "
"Install pyyaml to parse spec as yaml"
) from err
)
raise ValueError(msg) from err
else:
spec = yaml.load(cell, Loader=yaml.SafeLoader)

Expand Down
26 changes: 12 additions & 14 deletions altair/expr/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -184,51 +184,49 @@ def __getitem__(self, val):

class UnaryExpression(Expression):
def __init__(self, op, val):
super(UnaryExpression, self).__init__(op=op, val=val)
super().__init__(op=op, val=val)

def __repr__(self):
return "({op}{val})".format(op=self.op, val=_js_repr(self.val))
return f"({self.op}{_js_repr(self.val)})"


class BinaryExpression(Expression):
def __init__(self, op, lhs, rhs):
super(BinaryExpression, self).__init__(op=op, lhs=lhs, rhs=rhs)
super().__init__(op=op, lhs=lhs, rhs=rhs)

def __repr__(self):
return "({lhs} {op} {rhs})".format(
op=self.op, lhs=_js_repr(self.lhs), rhs=_js_repr(self.rhs)
)
return f"({_js_repr(self.lhs)} {self.op} {_js_repr(self.rhs)})"


class FunctionExpression(Expression):
def __init__(self, name, args):
super(FunctionExpression, self).__init__(name=name, args=args)
super().__init__(name=name, args=args)

def __repr__(self):
args = ",".join(_js_repr(arg) for arg in self.args)
return "{name}({args})".format(name=self.name, args=args)
return f"{self.name}({args})"


class ConstExpression(Expression):
def __init__(self, name, doc):
self.__doc__ = """{}: {}""".format(name, doc)
super(ConstExpression, self).__init__(name=name, doc=doc)
self.__doc__ = f"""{name}: {doc}"""
super().__init__(name=name, doc=doc)

def __repr__(self):
return str(self.name)


class GetAttrExpression(Expression):
def __init__(self, group, name):
super(GetAttrExpression, self).__init__(group=group, name=name)
super().__init__(group=group, name=name)

def __repr__(self):
return "{}.{}".format(self.group, self.name)
return f"{self.group}.{self.name}"


class GetItemExpression(Expression):
def __init__(self, group, name):
super(GetItemExpression, self).__init__(group=group, name=name)
super().__init__(group=group, name=name)

def __repr__(self):
return "{}[{!r}]".format(self.group, self.name)
return f"{self.group}[{self.name!r}]"
4 changes: 2 additions & 2 deletions altair/expr/funcs.py
Original file line number Diff line number Diff line change
Expand Up @@ -172,13 +172,13 @@ class ExprFunc:
def __init__(self, name, doc):
self.name = name
self.doc = doc
self.__doc__ = """{}(*args)\n {}""".format(name, doc)
self.__doc__ = f"""{name}(*args)\n {doc}"""

def __call__(self, *args):
return FunctionExpression(self.name, args)

def __repr__(self):
return "<function expr.{}(*args)>".format(self.name)
return f"<function expr.{self.name}(*args)>"


def _populate_namespace():
Expand Down
3 changes: 2 additions & 1 deletion altair/jupyter/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,14 +7,15 @@
# when anywidget is not installed
class JupyterChart:
def __init__(self, *args, **kwargs):
raise ImportError(
msg = (
"The Altair JupyterChart requires the anywidget \n"
"Python package which may be installed using pip with\n"
" pip install anywidget\n"
"or using conda with\n"
" conda install -c conda-forge anywidget\n"
"Afterwards, you will need to restart your Python kernel."
)
raise ImportError(msg)

else:
from .jupyter_chart import JupyterChart # noqa: F401
9 changes: 6 additions & 3 deletions altair/jupyter/jupyter_chart.py
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,8 @@ def __init__(self, trait_values):
elif isinstance(value, IntervalSelection):
traitlet_type = traitlets.Instance(IntervalSelection)
else:
raise ValueError(f"Unexpected selection type: {type(value)}")
msg = f"Unexpected selection type: {type(value)}"
raise ValueError(msg)

# Add the new trait.
self.add_traits(**{key: traitlet_type})
Expand All @@ -82,10 +83,11 @@ def _make_read_only(self, change):
"""
if change["name"] in self.traits() and change["old"] != change["new"]:
self._set_value(change["name"], change["old"])
raise ValueError(
msg = (
"Selections may not be set from Python.\n"
f"Attempted to set select: {change['name']}"
)
raise ValueError(msg)

def _set_value(self, key, value):
self.unobserve(self._make_read_only, names=key)
Expand Down Expand Up @@ -278,7 +280,8 @@ def _on_change_chart(self, change):
name=clean_name, value={}, store=[]
)
else:
raise ValueError(f"Unexpected selection type {select.type}")
msg = f"Unexpected selection type {select.type}"
raise ValueError(msg)
selection_watches.append(clean_name)
initial_vl_selections[clean_name] = {"value": None, "store": []}
else:
Expand Down
18 changes: 9 additions & 9 deletions altair/utils/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,17 +16,17 @@


__all__ = (
"infer_vegalite_type",
"AltairDeprecationWarning",
"PluginRegistry",
"SchemaBase",
"Undefined",
"display_traceback",
"infer_encoding_types",
"sanitize_dataframe",
"infer_vegalite_type",
"parse_shorthand",
"sanitize_arrow_table",
"sanitize_dataframe",
"spec_to_html",
"parse_shorthand",
"use_signature",
"update_nested",
"display_traceback",
"AltairDeprecationWarning",
"SchemaBase",
"Undefined",
"PluginRegistry",
"use_signature",
)
6 changes: 0 additions & 6 deletions altair/utils/_dfi_types.py
Original file line number Diff line number Diff line change
Expand Up @@ -76,7 +76,6 @@ def dtype(self) -> Dtype:
- Data types not included: complex, Arrow-style null, binary, decimal,
and nested (list, struct, map, union) dtypes.
"""
pass

# Have to use a generic Any return type as not all libraries who implement
# the dataframe interchange protocol implement the TypedDict that is usually
Expand All @@ -103,7 +102,6 @@ def describe_categorical(self) -> Any:

TBD: are there any other in-memory representations that are needed?
"""
pass


class DataFrame(Protocol):
Expand Down Expand Up @@ -136,19 +134,16 @@ def __dataframe__(
necessary if a library supports strided buffers, given that this protocol
specifies contiguous buffers.
"""
pass

def column_names(self) -> Iterable[str]:
"""
Return an iterator yielding the column names.
"""
pass

def get_column_by_name(self, name: str) -> Column:
"""
Return the column whose name is the indicated name.
"""
pass

def get_chunks(self, n_chunks: Optional[int] = None) -> Iterable["DataFrame"]:
"""
Expand All @@ -162,4 +157,3 @@ def get_chunks(self, n_chunks: Optional[int] = None) -> Iterable["DataFrame"]:
Note that the producer must ensure that all columns are chunked the
same way.
"""
pass
24 changes: 15 additions & 9 deletions altair/utils/_importers.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,18 +9,19 @@ def import_vegafusion() -> ModuleType:
version = importlib_version("vegafusion")
embed_version = importlib_version("vegafusion-python-embed")
if version != embed_version or Version(version) < Version(min_version):
raise RuntimeError(
msg = (
"The versions of the vegafusion and vegafusion-python-embed packages must match\n"
f"and must be version {min_version} or greater.\n"
f"Found:\n"
f" - vegafusion=={version}\n"
f" - vegafusion-python-embed=={embed_version}\n"
)
raise RuntimeError(msg)
import vegafusion as vf # type: ignore

return vf
except ImportError as err:
raise ImportError(
msg = (
'The "vegafusion" data transformer and chart.transformed_data feature requires\n'
f"version {min_version} or greater of the 'vegafusion-python-embed' and 'vegafusion' packages.\n"
"These can be installed with pip using:\n"
Expand All @@ -29,31 +30,34 @@ def import_vegafusion() -> ModuleType:
f' conda install -c conda-forge "vegafusion-python-embed>={min_version}" '
f'"vegafusion>={min_version}"\n\n'
f"ImportError: {err.args[0]}"
) from err
)
raise ImportError(msg) from err


def import_vl_convert() -> ModuleType:
min_version = "1.3.0"
try:
version = importlib_version("vl-convert-python")
if Version(version) < Version(min_version):
raise RuntimeError(
msg = (
f"The vl-convert-python package must be version {min_version} or greater. "
f"Found version {version}"
)
raise RuntimeError(msg)
import vl_convert as vlc

return vlc
except ImportError as err:
raise ImportError(
msg = (
f"The vl-convert Vega-Lite compiler and file export feature requires\n"
f"version {min_version} or greater of the 'vl-convert-python' package. \n"
f"This can be installed with pip using:\n"
f' pip install "vl-convert-python>={min_version}"\n'
"or conda:\n"
f' conda install -c conda-forge "vl-convert-python>={min_version}"\n\n'
f"ImportError: {err.args[0]}"
) from err
)
raise ImportError(msg) from err


def vl_version_for_vl_convert() -> str:
Expand All @@ -70,23 +74,25 @@ def import_pyarrow_interchange() -> ModuleType:
version = importlib_version("pyarrow")

if Version(version) < Version(min_version):
raise RuntimeError(
msg = (
f"The pyarrow package must be version {min_version} or greater. "
f"Found version {version}"
)
raise RuntimeError(msg)
import pyarrow.interchange as pi

return pi
except ImportError as err:
raise ImportError(
msg = (
f"Usage of the DataFrame Interchange Protocol requires\n"
f"version {min_version} or greater of the pyarrow package. \n"
f"This can be installed with pip using:\n"
f' pip install "pyarrow>={min_version}"\n'
"or conda:\n"
f' conda install -c conda-forge "pyarrow>={min_version}"\n\n'
f"ImportError: {err.args[0]}"
) from err
)
raise ImportError(msg) from err


def pyarrow_available() -> bool:
Expand Down
7 changes: 2 additions & 5 deletions altair/utils/_show.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,10 +26,7 @@ def open_html_in_browser(
Port to use. Defaults to a random port
"""
# Encode html to bytes
if isinstance(html, str):
html_bytes = html.encode("utf8")
else:
html_bytes = html
html_bytes = html.encode("utf8") if isinstance(html, str) else html

browser = None

Expand Down Expand Up @@ -69,5 +66,5 @@ def log_message(self, format, *args):
server = HTTPServer(
("127.0.0.1", port if port is not None else 0), OneShotRequestHandler
)
browser.open("http://127.0.0.1:%s" % server.server_port)
browser.open(f"http://127.0.0.1:{server.server_port}")
server.handle_request()
Loading
Loading