From 8e8cea17af44ba416c8b23e891ca165c75b76c0a Mon Sep 17 00:00:00 2001 From: William FH <13333726+hinthornw@users.noreply.github.com> Date: Wed, 22 May 2024 13:17:54 -0700 Subject: [PATCH] [Python] Add Null Sentry (#722) Previously, you couldn't write `expect(None).to_*` --- python/langsmith/_expect.py | 67 +++++++++++++++++++++----- python/pyproject.toml | 2 +- python/tests/unit_tests/test_expect.py | 20 ++++++++ 3 files changed, 76 insertions(+), 13 deletions(-) create mode 100644 python/tests/unit_tests/test_expect.py diff --git a/python/langsmith/_expect.py b/python/langsmith/_expect.py index db914c31e..75faa3f19 100644 --- a/python/langsmith/_expect.py +++ b/python/langsmith/_expect.py @@ -48,7 +48,15 @@ def test_output_semantically_close(): import atexit import concurrent.futures import inspect -from typing import TYPE_CHECKING, Any, Callable, Optional, Union, overload +from typing import ( + TYPE_CHECKING, + Any, + Callable, + Literal, + Optional, + Union, + overload, +) from langsmith import client as ls_client from langsmith import run_helpers as rh @@ -59,18 +67,34 @@ def test_output_semantically_close(): from langsmith._internal._embedding_distance import EmbeddingConfig +# Sentinel class used until PEP 0661 is accepted +class _NULL_SENTRY: + """A sentinel singleton class used to distinguish omitted keyword arguments + from those passed in with the value None (which may have different behavior). + """ # noqa: D205 + + def __bool__(self) -> Literal[False]: + return False + + def __repr__(self) -> str: + return "NOT_GIVEN" + + +NOT_GIVEN = _NULL_SENTRY() + + class _Matcher: """A class for making assertions on expectation values.""" def __init__( self, - client: ls_client.Client, + client: Optional[ls_client.Client], key: str, value: Any, _executor: Optional[concurrent.futures.ThreadPoolExecutor] = None, run_id: Optional[str] = None, ): - self.client = client + self._client = client self.key = key self.value = value self._executor = _executor or concurrent.futures.ThreadPoolExecutor( @@ -81,8 +105,10 @@ def __init__( def _submit_feedback(self, score: int, message: Optional[str] = None) -> None: if not ls_utils.test_tracking_is_disabled(): + if not self._client: + self._client = ls_client.Client() self._executor.submit( - self.client.create_feedback, + self._client.create_feedback, run_id=self._run_id, key="expectation", score=score, @@ -179,6 +205,18 @@ def to_equal(self, value: float) -> None: "to_equal", ) + def to_be_none(self) -> None: + """Assert that the expectation value is None. + + Raises: + AssertionError: If the expectation value is not None. + """ + self._assert( + self.value is None, + f"Expected {self.key} to be None, but got {self.value}", + "to_be_none", + ) + def to_contain(self, value: Any) -> None: """Assert that the expectation value contains the given value. @@ -216,7 +254,7 @@ class _Expect: """A class for setting expectations on test results.""" def __init__(self, *, client: Optional[ls_client.Client] = None): - self.client = client or ls_client.Client() + self._client = client self.executor = concurrent.futures.ThreadPoolExecutor(max_workers=3) atexit.register(self.executor.shutdown, wait=True) @@ -271,7 +309,7 @@ def embedding_distance( }, ) return _Matcher( - self.client, "embedding_distance", score, _executor=self.executor + self._client, "embedding_distance", score, _executor=self.executor ) def edit_distance( @@ -321,7 +359,7 @@ def edit_distance( }, ) return _Matcher( - self.client, + self._client, "edit_distance", score, _executor=self.executor, @@ -339,7 +377,7 @@ def value(self, value: Any) -> _Matcher: Examples: >>> expect.value(10).to_be_less_than(20) """ - return _Matcher(self.client, "value", value, _executor=self.executor) + return _Matcher(self._client, "value", value, _executor=self.executor) def score( self, @@ -370,7 +408,7 @@ def score( "comment": comment, }, ) - return _Matcher(self.client, key, score, _executor=self.executor) + return _Matcher(self._client, key, score, _executor=self.executor) ## Private Methods @@ -381,10 +419,13 @@ def __call__(self, value: Any, /) -> _Matcher: ... def __call__(self, /, *, client: ls_client.Client) -> _Expect: ... def __call__( - self, value: Optional[Any] = None, /, client: Optional[ls_client.Client] = None + self, + value: Optional[Any] = NOT_GIVEN, + /, + client: Optional[ls_client.Client] = None, ) -> Union[_Expect, _Matcher]: expected = _Expect(client=client) - if value is not None: + if value is not NOT_GIVEN: return expected.value(value) return expected @@ -392,8 +433,10 @@ def _submit_feedback(self, key: str, results: dict): current_run = rh.get_current_run_tree() run_id = current_run.trace_id if current_run else None if not ls_utils.test_tracking_is_disabled(): + if not self._client: + self._client = ls_client.Client() self.executor.submit( - self.client.create_feedback, run_id=run_id, key=key, **results + self._client.create_feedback, run_id=run_id, key=key, **results ) diff --git a/python/pyproject.toml b/python/pyproject.toml index 976f30004..f17acaf89 100644 --- a/python/pyproject.toml +++ b/python/pyproject.toml @@ -1,6 +1,6 @@ [tool.poetry] name = "langsmith" -version = "0.1.60" +version = "0.1.61" description = "Client library to connect to the LangSmith LLM Tracing and Evaluation Platform." authors = ["LangChain "] license = "MIT" diff --git a/python/tests/unit_tests/test_expect.py b/python/tests/unit_tests/test_expect.py new file mode 100644 index 000000000..cdf7ea9b2 --- /dev/null +++ b/python/tests/unit_tests/test_expect.py @@ -0,0 +1,20 @@ +from unittest import mock + +from langsmith import expect +from langsmith._expect import ls_client + + +def _is_none(x: object) -> bool: + return x is None + + +@mock.patch.object(ls_client, "Client", autospec=True) +def test_expect_explicit_none(mock_client: mock.Mock) -> None: + expect(None).against(_is_none) + expect(None).to_be_none() + expect.score(1).to_equal(1) + expect.score(1).to_be_less_than(2) + expect.score(1).to_be_greater_than(0) + expect.score(1).to_be_between(0, 2) + expect.score(1).to_be_approximately(1, 2) + expect({1, 2}).to_contain(1)