lmnr 0.6.19__py3-none-any.whl → 0.6.20__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -12,6 +12,7 @@ from lmnr.sdk.client.asynchronous.resources import (
12
12
  AsyncBrowserEvents,
13
13
  AsyncEvals,
14
14
  AsyncTags,
15
+ AsyncEvaluators
15
16
  )
16
17
  from lmnr.sdk.utils import from_env
17
18
 
@@ -74,6 +75,9 @@ class AsyncLaminarClient:
74
75
  self.__evals = AsyncEvals(
75
76
  self.__client, self.__base_url, self.__project_api_key
76
77
  )
78
+ self.__evaluators = AsyncEvaluators(
79
+ self.__client, self.__base_url, self.__project_api_key
80
+ )
77
81
  self.__browser_events = AsyncBrowserEvents(
78
82
  self.__client, self.__base_url, self.__project_api_key
79
83
  )
@@ -115,6 +119,15 @@ class AsyncLaminarClient:
115
119
  """
116
120
  return self.__tags
117
121
 
122
+ @property
123
+ def evaluators(self) -> AsyncEvaluators:
124
+ """Get the Evaluators resource.
125
+
126
+ Returns:
127
+ Evaluators: The Evaluators resource instance.
128
+ """
129
+ return self.__evaluators
130
+
118
131
  def is_closed(self) -> bool:
119
132
  return self.__client.is_closed
120
133
 
@@ -2,10 +2,12 @@ from lmnr.sdk.client.asynchronous.resources.agent import AsyncAgent
2
2
  from lmnr.sdk.client.asynchronous.resources.browser_events import AsyncBrowserEvents
3
3
  from lmnr.sdk.client.asynchronous.resources.evals import AsyncEvals
4
4
  from lmnr.sdk.client.asynchronous.resources.tags import AsyncTags
5
+ from lmnr.sdk.client.asynchronous.resources.evaluators import AsyncEvaluators
5
6
 
6
7
  __all__ = [
7
8
  "AsyncAgent",
8
9
  "AsyncEvals",
9
10
  "AsyncBrowserEvents",
10
11
  "AsyncTags",
12
+ "AsyncEvaluators"
11
13
  ]
@@ -0,0 +1,85 @@
1
+ """Evaluators resource for creating evaluator scores."""
2
+
3
+ import uuid
4
+ from typing import Any
5
+
6
+ from lmnr.sdk.client.asynchronous.resources.base import BaseAsyncResource
7
+ from lmnr.sdk.utils import format_id
8
+
9
+
10
+ class AsyncEvaluators(BaseAsyncResource):
11
+ """Resource for creating evaluator scores."""
12
+
13
+ async def score(
14
+ self,
15
+ *,
16
+ name: str,
17
+ trace_id: str | int | uuid.UUID | None = None,
18
+ span_id: str | int | uuid.UUID | None = None,
19
+ metadata: dict[str, Any] | None = None,
20
+ score: float,
21
+ ) -> None:
22
+ """Create a score for a span.
23
+
24
+ Args:
25
+ name (str): Name of the score
26
+ trace_id (str | int | uuid.UUID | None, optional): The trace ID to score (will be attached to root span)
27
+ span_id (str | int | uuid.UUID | None, optional): The span ID to score
28
+ metadata (dict[str, Any] | None, optional): Additional metadata. Defaults to None.
29
+ score (float): The score value (float)
30
+
31
+ Raises:
32
+ ValueError: If there's an error creating the score.
33
+
34
+ Example:
35
+ Score by trace ID (will attach to root span):
36
+
37
+ >>> await laminar_client.evaluators.score(
38
+ ... name="quality",
39
+ ... trace_id="trace-id-here",
40
+ ... score=0.95,
41
+ ... metadata={"model": "gpt-4"}
42
+ ... )
43
+
44
+ Score by span ID:
45
+
46
+ >>> await laminar_client.evaluators.score(
47
+ ... name="relevance",
48
+ ... span_id="span-id-here",
49
+ ... score=0.87
50
+ ... )
51
+ """
52
+ if trace_id is not None and span_id is not None:
53
+ raise ValueError("Cannot provide both trace_id and span_id. Please provide only one.")
54
+ if trace_id is None and span_id is None:
55
+ raise ValueError("Either 'trace_id' or 'span_id' must be provided.")
56
+
57
+ if trace_id is not None:
58
+ formatted_trace_id = format_id(trace_id)
59
+ payload = {
60
+ "name": name,
61
+ "traceId": formatted_trace_id,
62
+ "metadata": metadata,
63
+ "score": score,
64
+ "source": "Code",
65
+ }
66
+ else:
67
+ formatted_span_id = format_id(span_id)
68
+ payload = {
69
+ "name": name,
70
+ "spanId": formatted_span_id,
71
+ "metadata": metadata,
72
+ "score": score,
73
+ "source": "Code",
74
+ }
75
+
76
+ response = await self._client.post(
77
+ self._base_url + "/v1/evaluators/score",
78
+ json=payload,
79
+ headers=self._headers(),
80
+ )
81
+
82
+ if response.status_code != 200:
83
+ if response.status_code == 401:
84
+ raise ValueError("Unauthorized. Please check your project API key.")
85
+ raise ValueError(f"Error creating evaluator score: {response.text}")
@@ -5,6 +5,7 @@ import uuid
5
5
 
6
6
  from lmnr.sdk.client.asynchronous.resources.base import BaseAsyncResource
7
7
  from lmnr.sdk.log import get_default_logger
8
+ from lmnr.sdk.utils import format_id
8
9
 
9
10
  logger = get_default_logger(__name__)
10
11
 
@@ -54,18 +55,11 @@ class AsyncTags(BaseAsyncResource):
54
55
  ```
55
56
  """
56
57
  trace_tags = tags if isinstance(tags, list) else [tags]
57
- if isinstance(trace_id, uuid.UUID):
58
- trace_id = str(trace_id)
59
- elif isinstance(trace_id, int):
60
- trace_id = str(uuid.UUID(int=trace_id))
61
- elif isinstance(trace_id, str):
62
- uuid.UUID(trace_id) # Will raise ValueError if invalid
63
- else:
64
- raise ValueError(f"Invalid trace id: {trace_id}")
58
+ formatted_trace_id = format_id(trace_id)
65
59
 
66
60
  url = self._base_url + "/v1/tag"
67
61
  payload = {
68
- "traceId": trace_id,
62
+ "traceId": formatted_trace_id,
69
63
  "names": trace_tags,
70
64
  }
71
65
  response = await self._client.post(
@@ -78,7 +72,7 @@ class AsyncTags(BaseAsyncResource):
78
72
 
79
73
  if response.status_code == 404:
80
74
  logger.warning(
81
- f"Trace {trace_id} not found. The trace may have not been ended yet."
75
+ f"Trace {formatted_trace_id} not found. The trace may have not been ended yet."
82
76
  )
83
77
  return []
84
78
 
@@ -2,5 +2,6 @@ from lmnr.sdk.client.synchronous.resources.agent import Agent
2
2
  from lmnr.sdk.client.synchronous.resources.browser_events import BrowserEvents
3
3
  from lmnr.sdk.client.synchronous.resources.evals import Evals
4
4
  from lmnr.sdk.client.synchronous.resources.tags import Tags
5
+ from lmnr.sdk.client.synchronous.resources.evaluators import Evaluators
5
6
 
6
- __all__ = ["Agent", "Evals", "BrowserEvents", "Tags"]
7
+ __all__ = ["Agent", "Evals", "Evaluators", "BrowserEvents", "Tags"]
@@ -0,0 +1,85 @@
1
+ """Evaluators resource for creating evaluator scores."""
2
+
3
+ import uuid
4
+ from typing import Any
5
+
6
+ from lmnr.sdk.client.synchronous.resources.base import BaseResource
7
+ from lmnr.sdk.utils import format_id
8
+
9
+
10
+ class Evaluators(BaseResource):
11
+ """Resource for creating evaluator scores."""
12
+
13
+ def score(
14
+ self,
15
+ *,
16
+ name: str,
17
+ trace_id: str | int | uuid.UUID | None = None,
18
+ span_id: str | int | uuid.UUID | None = None,
19
+ metadata: dict[str, Any] | None = None,
20
+ score: float,
21
+ ) -> None:
22
+ """Create a score for a span.
23
+
24
+ Args:
25
+ name (str): Name of the score
26
+ trace_id (str | int | uuid.UUID | None, optional): The trace ID to score (will be attached to root span)
27
+ span_id (str | int | uuid.UUID | None, optional): The span ID to score
28
+ metadata (dict[str, Any] | None, optional): Additional metadata. Defaults to None.
29
+ score (float): The score value (float)
30
+
31
+ Raises:
32
+ ValueError: If there's an error creating the score.
33
+
34
+ Example:
35
+ Score by trace ID (will attach to root span):
36
+
37
+ >>> laminar_client.evaluators.score(
38
+ ... name="quality",
39
+ ... trace_id="trace-id-here",
40
+ ... score=0.95,
41
+ ... metadata={"model": "gpt-4"}
42
+ ... )
43
+
44
+ Score by span ID:
45
+
46
+ >>> laminar_client.evaluators.score(
47
+ ... name="relevance",
48
+ ... span_id="span-id-here",
49
+ ... score=0.87
50
+ ... )
51
+ """
52
+ if trace_id is not None and span_id is not None:
53
+ raise ValueError("Cannot provide both trace_id and span_id. Please provide only one.")
54
+ if trace_id is None and span_id is None:
55
+ raise ValueError("Either 'trace_id' or 'span_id' must be provided.")
56
+
57
+ if trace_id is not None:
58
+ formatted_trace_id = format_id(trace_id)
59
+ payload = {
60
+ "name": name,
61
+ "traceId": formatted_trace_id,
62
+ "metadata": metadata,
63
+ "score": score,
64
+ "source": "Code",
65
+ }
66
+ else:
67
+ formatted_span_id = format_id(span_id)
68
+ payload = {
69
+ "name": name,
70
+ "spanId": formatted_span_id,
71
+ "metadata": metadata,
72
+ "score": score,
73
+ "source": "Code",
74
+ }
75
+
76
+ response = self._client.post(
77
+ self._base_url + "/v1/evaluators/score",
78
+ json=payload,
79
+ headers=self._headers(),
80
+ )
81
+
82
+ if response.status_code != 200:
83
+ if response.status_code == 401:
84
+ raise ValueError("Unauthorized. Please check your project API key.")
85
+ raise ValueError(f"Error creating evaluator score: {response.text}")
@@ -5,6 +5,7 @@ import uuid
5
5
 
6
6
  from lmnr.sdk.client.synchronous.resources.base import BaseResource
7
7
  from lmnr.sdk.log import get_default_logger
8
+ from lmnr.sdk.utils import format_id
8
9
 
9
10
  logger = get_default_logger(__name__)
10
11
 
@@ -54,18 +55,11 @@ class Tags(BaseResource):
54
55
  ```
55
56
  """
56
57
  trace_tags = tags if isinstance(tags, list) else [tags]
57
- if isinstance(trace_id, uuid.UUID):
58
- trace_id = str(trace_id)
59
- elif isinstance(trace_id, int):
60
- trace_id = str(uuid.UUID(int=trace_id))
61
- elif isinstance(trace_id, str):
62
- uuid.UUID(trace_id)
63
- else:
64
- raise ValueError(f"Invalid trace id: {trace_id}")
58
+ formatted_trace_id = format_id(trace_id)
65
59
 
66
60
  url = self._base_url + "/v1/tag"
67
61
  payload = {
68
- "traceId": trace_id,
62
+ "traceId": formatted_trace_id,
69
63
  "names": trace_tags,
70
64
  }
71
65
  response = self._client.post(
@@ -78,7 +72,7 @@ class Tags(BaseResource):
78
72
 
79
73
  if response.status_code == 404:
80
74
  logger.warning(
81
- f"Trace {trace_id} not found. The trace may have not been ended yet."
75
+ f"Trace {formatted_trace_id} not found. The trace may have not been ended yet."
82
76
  )
83
77
  return []
84
78
 
@@ -11,6 +11,7 @@ from lmnr.sdk.client.synchronous.resources import (
11
11
  Agent,
12
12
  BrowserEvents,
13
13
  Evals,
14
+ Evaluators,
14
15
  Tags,
15
16
  )
16
17
  from lmnr.sdk.utils import from_env
@@ -27,6 +28,7 @@ class LaminarClient:
27
28
  __agent: Agent | None = None
28
29
  __evals: Evals | None = None
29
30
  __tags: Tags | None = None
31
+ __evaluators: Evaluators | None = None
30
32
 
31
33
  def __init__(
32
34
  self,
@@ -74,6 +76,9 @@ class LaminarClient:
74
76
  # Initialize resource objects
75
77
  self.__agent = Agent(self.__client, self.__base_url, self.__project_api_key)
76
78
  self.__evals = Evals(self.__client, self.__base_url, self.__project_api_key)
79
+ self.__evaluators = Evaluators(
80
+ self.__client, self.__base_url, self.__project_api_key
81
+ )
77
82
  self.__browser_events = BrowserEvents(
78
83
  self.__client, self.__base_url, self.__project_api_key
79
84
  )
@@ -115,6 +120,15 @@ class LaminarClient:
115
120
  """
116
121
  return self.__tags
117
122
 
123
+ @property
124
+ def evaluators(self) -> Evaluators:
125
+ """Get the Evaluators resource.
126
+
127
+ Returns:
128
+ Evaluators: The Evaluators resource instance.
129
+ """
130
+ return self.__evaluators
131
+
118
132
  def shutdown(self):
119
133
  """Shutdown the client by closing underlying connections."""
120
134
  self.__client.close()
lmnr/sdk/utils.py CHANGED
@@ -128,3 +128,26 @@ def is_otel_attribute_value_type(value: typing.Any) -> bool:
128
128
  )
129
129
  return True
130
130
  return False
131
+
132
+
133
+ def format_id(id_value: str | int | uuid.UUID) -> str:
134
+ """Format trace/span/evaluation ID to a UUID string, or return valid UUID strings as-is.
135
+
136
+ Args:
137
+ id_value: The ID in various formats (UUID, int, or valid UUID string)
138
+
139
+ Returns:
140
+ str: UUID string representation
141
+
142
+ Raises:
143
+ ValueError: If id_value cannot be converted to a valid UUID
144
+ """
145
+ if isinstance(id_value, uuid.UUID):
146
+ return str(id_value)
147
+ elif isinstance(id_value, int):
148
+ return str(uuid.UUID(int=id_value))
149
+ elif isinstance(id_value, str):
150
+ uuid.UUID(id_value)
151
+ return id_value
152
+ else:
153
+ raise ValueError(f"Invalid ID type: {type(id_value)}")
lmnr/version.py CHANGED
@@ -3,7 +3,7 @@ import httpx
3
3
  from packaging import version
4
4
 
5
5
 
6
- __version__ = "0.6.19"
6
+ __version__ = "0.6.20"
7
7
  PYTHON_VERSION = f"{sys.version_info.major}.{sys.version_info.minor}"
8
8
 
9
9
 
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: lmnr
3
- Version: 0.6.19
3
+ Version: 0.6.20
4
4
  Summary: Python SDK for Laminar
5
5
  Author: lmnr.ai
6
6
  Author-email: lmnr.ai <founders@lmnr.ai>
@@ -49,20 +49,22 @@ lmnr/sdk/browser/playwright_otel.py,sha256=59f477e921e777d49ccda4ef1017a50bb5584
49
49
  lmnr/sdk/browser/pw_utils.py,sha256=81bb14ffc53c47a72961e8d1e231efc473f9b2cb7e2992e97751e8dad0e69184,11009
50
50
  lmnr/sdk/browser/rrweb/rrweb.umd.min.cjs,sha256=2f2da38b00bb85312d8225f3069a09352135564ceed2564f652f1bae3bac016d,260896
51
51
  lmnr/sdk/browser/utils.py,sha256=4a668776d2938108d25fbcecd61c8e1710a4da3e56230d5fefca5964dd09e3c1,2371
52
- lmnr/sdk/client/asynchronous/async_client.py,sha256=da621ecd403352baa4e6d0999737e4203e0dcc8cbf26ec8142d0c2013fb632ba,4585
53
- lmnr/sdk/client/asynchronous/resources/__init__.py,sha256=f5f923955252f338670932138e1a30d7bde984175997289a7be5f42dc4d25aa3,381
52
+ lmnr/sdk/client/asynchronous/async_client.py,sha256=e8feae007506cd2e4b08e72706f5f1bb4ea54492b4aa6b68ef184a129de8f466,4948
53
+ lmnr/sdk/client/asynchronous/resources/__init__.py,sha256=993423ea462aa8ea37d8d91662341c1ca0711cb2447cd476aacc373858f76135,481
54
54
  lmnr/sdk/client/asynchronous/resources/agent.py,sha256=3a78372b62912cdeda831d7ff9a671306713fce185dff646b452e6f1a3cc6d8c,17788
55
55
  lmnr/sdk/client/asynchronous/resources/base.py,sha256=689e37435ae5b60db7210688e1e79a64a724c554e00d46c226b0a18500941281,986
56
56
  lmnr/sdk/client/asynchronous/resources/browser_events.py,sha256=4fe0d46db01f310d95aa255f815a65c6e4da259ba835c0b53ba442c5d7f0ed44,1163
57
57
  lmnr/sdk/client/asynchronous/resources/evals.py,sha256=8c6f8096916657ef269463b2d0585795d9cedad056a047abcde6365ff0b320bd,5761
58
- lmnr/sdk/client/asynchronous/resources/tags.py,sha256=55bb01329d76d1dffc76b185af539ba78c52464b733c2fb790e61c6e5667bca0,2565
59
- lmnr/sdk/client/synchronous/resources/__init__.py,sha256=8431b234045d1b7276e652c03fc26794447bafc24bf8942e3cdd717617a20b0e,318
58
+ lmnr/sdk/client/asynchronous/resources/evaluators.py,sha256=964046f5146e89032fbb701b883f4f3a7cb996aeb9ff368f86e8f967df2fef10,2918
59
+ lmnr/sdk/client/asynchronous/resources/tags.py,sha256=14fc2e38cae2f6fe126dc8dca085d7ad02d8d7c1a09bc4b5b5b8e38a0edf7348,2314
60
+ lmnr/sdk/client/synchronous/resources/__init__.py,sha256=685792a8c8494ea061592b86cb63d6bb0dca8d9848181aa11b7d97d5714df337,403
60
61
  lmnr/sdk/client/synchronous/resources/agent.py,sha256=9a74eeeada0dd8b6e0984850fa6759d02ccd02792b1a292caf2b34032330cf60,17809
61
62
  lmnr/sdk/client/synchronous/resources/base.py,sha256=9ded59675d1498d90cac4095bc295c1097dc1499521af697382f0aea66533dd6,971
62
63
  lmnr/sdk/client/synchronous/resources/browser_events.py,sha256=f6b1585997ac5d0a269c581b679f74b4614c4da363d0e0334fd45c1700fcabf6,1135
63
64
  lmnr/sdk/client/synchronous/resources/evals.py,sha256=415fefe234519f8affb24d858efa9d6c0735f966b6194977a96ac2ce16d066c0,7008
64
- lmnr/sdk/client/synchronous/resources/tags.py,sha256=70d304ccc0e194136923b278c7ac6414000d88d4aaf95baeff38b934f93691c0,2485
65
- lmnr/sdk/client/synchronous/sync_client.py,sha256=208ce3fa60301c7a11b945fd2a426daf34c68b950eca06c0f30880f40ab37f61,4907
65
+ lmnr/sdk/client/synchronous/resources/evaluators.py,sha256=3cd6a17e7a9cc0441c2d20bf6cf46ce3720131cc30053e2cd124e5668c75f49a,2879
66
+ lmnr/sdk/client/synchronous/resources/tags.py,sha256=123deec43128662c21cb275b2df6a102372f875315b0bd36806555394c1d4b5b,2270
67
+ lmnr/sdk/client/synchronous/sync_client.py,sha256=0bebe88e3aed689505e9ed3d32036f76df4c3496e4d659162bd41abedc026f16,5299
66
68
  lmnr/sdk/datasets.py,sha256=3fd851c5f97bf88eaa84b1451a053eaff23b4497cbb45eac2f9ea0e5f2886c00,1708
67
69
  lmnr/sdk/decorators.py,sha256=d6ebbdc7105881b945aa5850a87d7b71962aedb69b026075944b6f0604ec3f41,4468
68
70
  lmnr/sdk/eval_control.py,sha256=291394ac385c653ae9b5167e871bebeb4fe8fc6b7ff2ed38e636f87015dcba86,184
@@ -70,9 +72,9 @@ lmnr/sdk/evaluations.py,sha256=ad40c6a48d5e34aa33f3d9ad80514f2055e7eb360b820697b
70
72
  lmnr/sdk/laminar.py,sha256=a0e55ca3f73d66cb53ef51ecaae1ac81bb45ba65ddd848f3d2b97faa998c9535,33996
71
73
  lmnr/sdk/log.py,sha256=9edfd83263f0d4845b1b2d1beeae2b4ed3f8628de941f371a893d72b79c348d4,2213
72
74
  lmnr/sdk/types.py,sha256=650a7949e609359b0adcaadb49e5cf63fc67ea61a35b9992c3f8b445dfce6b89,12328
73
- lmnr/sdk/utils.py,sha256=cab707221a000dff65587f6a25664c9a445662f7740eec4f48b3f7998e98170d,4327
74
- lmnr/version.py,sha256=e1e5a6ad24ee26cbf40f635c206464fb6dd4906e4b8881e03c451796b4b5f843,1322
75
- lmnr-0.6.19.dist-info/WHEEL,sha256=66530aef82d5020ef5af27ae0123c71abb9261377c5bc519376c671346b12918,79
76
- lmnr-0.6.19.dist-info/entry_points.txt,sha256=abdf3411b7dd2d7329a241f2da6669bab4e314a747a586ecdb9f888f3035003c,39
77
- lmnr-0.6.19.dist-info/METADATA,sha256=ecbff9a35e6ae55ded780b747c97344d434f0f4617cf3caf6545fc367f5b932e,14475
78
- lmnr-0.6.19.dist-info/RECORD,,
75
+ lmnr/sdk/utils.py,sha256=4beb884ae6fbbc7d8cf639b036b726ea6a2a658f0a6386faf5735a13d706a2d8,5039
76
+ lmnr/version.py,sha256=fec940af89e4f538f3f2c79e3670aaf085c8fa8edd10da2a292bcd224a3f2a26,1322
77
+ lmnr-0.6.20.dist-info/WHEEL,sha256=66530aef82d5020ef5af27ae0123c71abb9261377c5bc519376c671346b12918,79
78
+ lmnr-0.6.20.dist-info/entry_points.txt,sha256=abdf3411b7dd2d7329a241f2da6669bab4e314a747a586ecdb9f888f3035003c,39
79
+ lmnr-0.6.20.dist-info/METADATA,sha256=dd5758deaf108a40ec4b216482914e90f414199c716887e94af4328b74a32878,14475
80
+ lmnr-0.6.20.dist-info/RECORD,,
File without changes