arize-phoenix 5.7.0__py3-none-any.whl → 5.8.0__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.

Potentially problematic release.


This version of arize-phoenix might be problematic. Click here for more details.

Files changed (24) hide show
  1. {arize_phoenix-5.7.0.dist-info → arize_phoenix-5.8.0.dist-info}/METADATA +3 -5
  2. {arize_phoenix-5.7.0.dist-info → arize_phoenix-5.8.0.dist-info}/RECORD +24 -24
  3. {arize_phoenix-5.7.0.dist-info → arize_phoenix-5.8.0.dist-info}/WHEEL +1 -1
  4. phoenix/config.py +19 -3
  5. phoenix/server/api/helpers/playground_clients.py +123 -36
  6. phoenix/server/api/helpers/playground_spans.py +173 -76
  7. phoenix/server/api/input_types/InvocationParameters.py +7 -8
  8. phoenix/server/api/mutations/chat_mutations.py +46 -65
  9. phoenix/server/api/subscriptions.py +210 -158
  10. phoenix/server/api/types/ChatCompletionSubscriptionPayload.py +5 -3
  11. phoenix/server/app.py +14 -0
  12. phoenix/server/grpc_server.py +3 -1
  13. phoenix/server/static/.vite/manifest.json +31 -31
  14. phoenix/server/static/assets/{components-Csu8UKOs.js → components-MllbfxfJ.js} +168 -150
  15. phoenix/server/static/assets/{index-Bk5C9EA7.js → index-BVO2YcT1.js} +2 -2
  16. phoenix/server/static/assets/{pages-UeWaKXNs.js → pages-BHfC6jnL.js} +394 -300
  17. phoenix/server/static/assets/{vendor-CtqfhlbC.js → vendor-BEuNhfwH.js} +1 -1
  18. phoenix/server/static/assets/{vendor-arizeai-C_3SBz56.js → vendor-arizeai-Bskhzyjm.js} +1 -1
  19. phoenix/server/static/assets/{vendor-codemirror-wfdk9cjp.js → vendor-codemirror-DLlXCf0x.js} +1 -1
  20. phoenix/server/static/assets/{vendor-recharts-BiVnSv90.js → vendor-recharts-CRqhvLYg.js} +1 -1
  21. phoenix/version.py +1 -1
  22. {arize_phoenix-5.7.0.dist-info → arize_phoenix-5.8.0.dist-info}/entry_points.txt +0 -0
  23. {arize_phoenix-5.7.0.dist-info → arize_phoenix-5.8.0.dist-info}/licenses/IP_NOTICE +0 -0
  24. {arize_phoenix-5.7.0.dist-info → arize_phoenix-5.8.0.dist-info}/licenses/LICENSE +0 -0
@@ -1,14 +1,12 @@
1
1
  Metadata-Version: 2.3
2
2
  Name: arize-phoenix
3
- Version: 5.7.0
3
+ Version: 5.8.0
4
4
  Summary: AI Observability and Evaluation
5
5
  Project-URL: Documentation, https://docs.arize.com/phoenix/
6
6
  Project-URL: Issues, https://github.com/Arize-ai/phoenix/issues
7
7
  Project-URL: Source, https://github.com/Arize-ai/phoenix
8
8
  Author-email: Arize AI <phoenix-devs@arize.com>
9
9
  License: Elastic-2.0
10
- License-File: IP_NOTICE
11
- License-File: LICENSE
12
10
  Keywords: Explainability,Monitoring,Observability
13
11
  Classifier: Programming Language :: Python
14
12
  Classifier: Programming Language :: Python :: 3.9
@@ -131,7 +129,7 @@ Description-Content-Type: text/markdown
131
129
  <a href="https://docs.arize.com/phoenix/">
132
130
  <img src="https://img.shields.io/static/v1?message=Docs&logo=&labelColor=grey&color=blue&logoColor=white&label=%20"/>
133
131
  </a>
134
- <a target="_blank" href="https://join.slack.com/t/arize-ai/shared_invite/zt-1px8dcmlf-fmThhDFD_V_48oU7ALan4Q">
132
+ <a target="_blank" href="https://join.slack.com/t/arize-ai/shared_invite/zt-22vj03k4k-MlrNEwv5WeswapTs0kNCBw">
135
133
  <img src="https://img.shields.io/static/v1?message=Community&logo=slack&labelColor=grey&color=blue&logoColor=white&label=%20"/>
136
134
  </a>
137
135
  <a target="_blank" href="https://twitter.com/ArizePhoenix">
@@ -237,7 +235,7 @@ For details about tracing integrations and example applications, see the [OpenIn
237
235
 
238
236
  Join our community to connect with thousands of AI builders.
239
237
 
240
- - 🌍 Join our [Slack community](https://join.slack.com/t/arize-ai/shared_invite/zt-1px8dcmlf-fmThhDFD_V_48oU7ALan4Q).
238
+ - 🌍 Join our [Slack community](https://join.slack.com/t/arize-ai/shared_invite/zt-22vj03k4k-MlrNEwv5WeswapTs0kNCBw).
241
239
  - 📚 Read our [documentation](https://docs.arize.com/phoenix).
242
240
  - 💡 Ask questions and provide feedback in the _#phoenix-support_ channel.
243
241
  - 🌟 Leave a star on our [GitHub](https://github.com/Arize-ai/phoenix).
@@ -1,12 +1,12 @@
1
1
  phoenix/__init__.py,sha256=X3eUEwd2rG8KKWWYVNNDJoqo08ihfjgHhlP29dcdNJE,5481
2
2
  phoenix/auth.py,sha256=JpkwJbis2INlIXWcQ-M_Nk5Ln9LBgHMdWNnaAQp0D2w,10940
3
- phoenix/config.py,sha256=y-JvQJRtxO2BSrXdy4d6CSkpzV9dmfT6hKSUiWpBRws,24282
3
+ phoenix/config.py,sha256=wmHEfX2npXWSRtTR9-bz8gJUbfacjcs6w7DDSkvosro,24902
4
4
  phoenix/datetime_utils.py,sha256=iJzNG6YJ6V7_u8B2iA7P2Z26FyxYbOPtx0dhJ7kNDHA,3398
5
5
  phoenix/exceptions.py,sha256=n2L2KKuecrdflB9MsCdAYCiSEvGJptIsfRkXMoJle7A,169
6
6
  phoenix/py.typed,sha256=AbpHGcgLb-kRsJGnwFEktk7uzpZOCcBY74-YBdrKVGs,1
7
7
  phoenix/services.py,sha256=kpW1WL0kiB8XJsO6XycvZVJ-lBkNoenhQ7atCvBoSe8,5365
8
8
  phoenix/settings.py,sha256=ht-0oN-sMV6SPXrk7Tu1EZlngpAYkGNLYPhO8DyrdQI,661
9
- phoenix/version.py,sha256=QmHMXVnw5DVPfWzvN7FS1tOhDAesdxpM_aVOh9CMuSk,22
9
+ phoenix/version.py,sha256=duDTiv1rL8Ee9_jtzzhpq-j4xkkpVPkUh2Daa_Ou-xA,22
10
10
  phoenix/core/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
11
11
  phoenix/core/embedding_dimension.py,sha256=zKGbcvwOXgLf-yrJBpQyKtd-LEOPRKHnUToyAU8Owis,87
12
12
  phoenix/core/model.py,sha256=qBFraOtmwCCnWJltKNP18DDG0mULXigytlFsa6YOz6k,4837
@@ -73,11 +73,11 @@ phoenix/pointcloud/pointcloud.py,sha256=SN_1wXZcwKrtSnHGZLDZGx71orqE1WyVF7E-D58d
73
73
  phoenix/pointcloud/projectors.py,sha256=TQgwc9cJDjJkin1WZyZzgl3HsYrLLiyWD7Czy4jNW3U,1088
74
74
  phoenix/pointcloud/umap_parameters.py,sha256=db_WEPoamuWtopZx7tQfAXPnoE0MS8FkAV0_ThjEx_Q,1735
75
75
  phoenix/server/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
76
- phoenix/server/app.py,sha256=feZL0fpIS6QuJQmoDl3yDhsDrl4-bsnhJfR3XuHxguQ,37247
76
+ phoenix/server/app.py,sha256=Nh1CR4YWXnQYULMJuLQPdz4VVt7nwz5E8IDHtR91_GQ,37847
77
77
  phoenix/server/bearer_auth.py,sha256=0UudvkAS_dxna5JEJJhGUYwB6Ny-e22ssX5Mm79QwCk,5907
78
78
  phoenix/server/dml_event.py,sha256=MjJmVEKytq75chBOSyvYDusUnEbg1pHpIjR3pZkUaJA,2838
79
79
  phoenix/server/dml_event_handler.py,sha256=EZLXmCvx4pJrCkz29gxwKwmvmUkTtPCHw6klR-XM8qE,8258
80
- phoenix/server/grpc_server.py,sha256=42KsTfHQiTnf1G9KKjn04I7KOjZXeamc4sMClwumMOk,3754
80
+ phoenix/server/grpc_server.py,sha256=SknR-iLIUqU9swiAyLhbCUREA1obOM6w49xgdK1AQDs,3839
81
81
  phoenix/server/jwt_store.py,sha256=asxzY4_ZBM2FWAMstHvhvnKUP_0AA3v3xPTL2IOgNqY,16831
82
82
  phoenix/server/main.py,sha256=cJaJArrLof5ziFkpB_31TGTaiUK_YCh7_ROZRtrdZ-Q,16310
83
83
  phoenix/server/oauth2.py,sha256=EV4wcCwG0N7cJRcfGNURdP5rZgRVCeRDvXyle19A27Y,2064
@@ -94,7 +94,7 @@ phoenix/server/api/exceptions.py,sha256=TA0JuY2YRnj35qGuMSQ8d0ToHum9gWm9W--3fSKH
94
94
  phoenix/server/api/interceptor.py,sha256=ykDnoC_apUd-llVli3m1CW18kNSIgjz2qZ6m5JmPDu8,1294
95
95
  phoenix/server/api/queries.py,sha256=nOgrAygpl49y1a5V0O_y4SRNJ8dwgv_C7birrgkgRm4,27150
96
96
  phoenix/server/api/schema.py,sha256=tHyw2jTbue_-gu0fe9Sw7LUYtzJUCwp9SvccDgOkNPw,1696
97
- phoenix/server/api/subscriptions.py,sha256=o8uepfk9ofh4jsWQ5wbbiOJHcSkZozizNgF_9UtD6Pc,17323
97
+ phoenix/server/api/subscriptions.py,sha256=Xu1k-UL893Ao1FsGk7OGfxuuyO6A3FzbSK8a2TTSDe4,19692
98
98
  phoenix/server/api/utils.py,sha256=quCBRcusc6PUq9tJq7M8PgwFZp7nXgVAxtbw8feribY,833
99
99
  phoenix/server/api/dataloaders/__init__.py,sha256=jNYvfXjnZzgA2HWTG7AZdqWGla3ZysBUDUei8Zkz6N8,3290
100
100
  phoenix/server/api/dataloaders/annotation_summaries.py,sha256=2sHmIDX7n8tuPeBTs9bMKtlMKWn_Ph9awTZqmwn2Owc,5505
@@ -125,9 +125,9 @@ phoenix/server/api/dataloaders/cache/__init__.py,sha256=SYoOM9n8FJaMdQarma5d1blu
125
125
  phoenix/server/api/dataloaders/cache/two_tier_cache.py,sha256=cmo8FUT3E91R139IEzh4yCga-6nTamc5KPXAfMrzNDM,2315
126
126
  phoenix/server/api/helpers/__init__.py,sha256=m2-xaSPqUiSs91k62JaRDjFNfl-1byxBfY-m_Vxw16U,272
127
127
  phoenix/server/api/helpers/dataset_helpers.py,sha256=14mldZp9to3rr9BdvvoFqEwZHHV_k2e7jPm8q9z2OdQ,6896
128
- phoenix/server/api/helpers/playground_clients.py,sha256=OGwoZfUX5RjdRoWD-_4Nr9wQocQcdOChDNn7riW5AX0,24641
128
+ phoenix/server/api/helpers/playground_clients.py,sha256=tgf94esqWmt1AnXEA7QDa2G3THjvtbbS_qLY5xyJd70,29093
129
129
  phoenix/server/api/helpers/playground_registry.py,sha256=CPLMziFB2wmr-dfbx7VbzO2f8YIG_k5RftzvGXYGQ1w,2570
130
- phoenix/server/api/helpers/playground_spans.py,sha256=MAB52fbVLffBnz0QxJw8J8Y_heuzbK_blfg6CUrnDxs,11761
130
+ phoenix/server/api/helpers/playground_spans.py,sha256=j7WlFTolCgrpkazwVuXqvTj8JSJ-nf-Gf_VZKCbzAvU,14775
131
131
  phoenix/server/api/input_types/AddExamplesToDatasetInput.py,sha256=mIQz0S_z8YdrktKIY6RCvtNJ2yZF9pYvTGgasUsI-54,430
132
132
  phoenix/server/api/input_types/AddSpansToDatasetInput.py,sha256=-StSstyMAVrba3tG1U30b-srkKCtu_svflQuSM19iJA,362
133
133
  phoenix/server/api/input_types/ChatCompletionInput.py,sha256=g_5ARuwylt-uCVAsGyZPEVtidEQiOhbKakvDQsZumzw,1451
@@ -150,7 +150,7 @@ phoenix/server/api/input_types/DimensionFilter.py,sha256=eBYcn7ECSJQlEePvbStqkHB
150
150
  phoenix/server/api/input_types/DimensionInput.py,sha256=Vfx5FmiMKey4-EHDQsQRPzSAMRJMN5oVMLDUl4NKAa8,164
151
151
  phoenix/server/api/input_types/GenerativeModelInput.py,sha256=h_9dNkz-LBgOLKQ5_ijch4UNGiDb4x5CCC96WyverSg,551
152
152
  phoenix/server/api/input_types/Granularity.py,sha256=PiOFWmSMs_w7ZZMNfEWPzIY6S6guI83Vy4NXXSEewGo,2310
153
- phoenix/server/api/input_types/InvocationParameters.py,sha256=OA4xC2t33alV1Y57zeIgQImUEI_EHTl4vRLYRtjyNTM,5186
153
+ phoenix/server/api/input_types/InvocationParameters.py,sha256=Lg6-SKsx72Lb-jzjJLSfPfmyt8bPz_8JtLpv5-TDcZQ,5154
154
154
  phoenix/server/api/input_types/PatchAnnotationInput.py,sha256=NWhkcbcGNPwfOYsN3wm5YFNNrSc5T-8Y5my74RK99HE,520
155
155
  phoenix/server/api/input_types/PatchDatasetExamplesInput.py,sha256=_uMqkAInhLDvzUSASl6HgLNulTsekMcYzyd5J6LF90I,884
156
156
  phoenix/server/api/input_types/PatchDatasetInput.py,sha256=OURtTVY8Z_oFEDtKwT1LCMaOK5D4QYo5TVQ6mDrex-g,328
@@ -164,7 +164,7 @@ phoenix/server/api/input_types/UserRoleInput.py,sha256=xxhFe0ITZOgRVEJbVem_W6F1I
164
164
  phoenix/server/api/input_types/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
165
165
  phoenix/server/api/mutations/__init__.py,sha256=1wClieLNA3_Tin4Ah67rkrQvwSSZAdPU0EPsRiUxyAA,1103
166
166
  phoenix/server/api/mutations/api_key_mutations.py,sha256=OmPipsmlQIb6DKvAgO58mZUwkYJihlJB2N4lTyeUlAA,6164
167
- phoenix/server/api/mutations/chat_mutations.py,sha256=eaZcuiweiXb59ZstpkFO7SmTvVwV4H1IpRLvc-zTvko,13898
167
+ phoenix/server/api/mutations/chat_mutations.py,sha256=MseQk4FEQOjEs-CseGDcD8Qf73GfKsDvcpKjwD4V9A4,13123
168
168
  phoenix/server/api/mutations/dataset_mutations.py,sha256=siwsSmozKRIK8ZhPrfWl-GxKCL4lAmRPms862tG2KXY,27064
169
169
  phoenix/server/api/mutations/experiment_mutations.py,sha256=p3CoLAa8nFPa3D759Y2A7De_PVJNGOL98mA3HoZBrRQ,3188
170
170
  phoenix/server/api/mutations/export_events_mutations.py,sha256=xoDnVWC7eA_8wNQP0-oyiHojyUZ0EhVVSrsAnztetC0,3993
@@ -196,7 +196,7 @@ phoenix/server/api/types/AnnotatorKind.py,sha256=rPgGdbN1Gvc109sGQ_ZH-gfJbp93V9w
196
196
  phoenix/server/api/types/ApiKey.py,sha256=76HIkWxyJqL3UjdMesX9l5y6XaD9QphwabXGQVY-MaQ,793
197
197
  phoenix/server/api/types/AuthMethod.py,sha256=M25usT6FCNKi-QFMdtMqRjJVGfq1ACnVoSjHXGxfrUk,126
198
198
  phoenix/server/api/types/ChatCompletionMessageRole.py,sha256=kmQOilOlVntlmqbaHkqXZuimfljfYaHc3kbo53uK8_0,227
199
- phoenix/server/api/types/ChatCompletionSubscriptionPayload.py,sha256=IhpidX4CvWGi4nOLB2BHhueObTp0-wVLHTfxZEm1HQA,919
199
+ phoenix/server/api/types/ChatCompletionSubscriptionPayload.py,sha256=G0YsirJzDMFDqEK7I_FCkPmIugAof-8D9fNfFlHdhjY,1031
200
200
  phoenix/server/api/types/Cluster.py,sha256=duX0pSC7P3YT1IztduuJtPsRj1x6oOOLfmWf2Y-FdEk,5699
201
201
  phoenix/server/api/types/CreateDatasetPayload.py,sha256=R-6zCmuD0f76RU9Giu78xwTHlASQs6Aq8yzvX1Kxc3g,140
202
202
  phoenix/server/api/types/DataQualityMetric.py,sha256=zRKsNvHBu-NdcsunuLhqFpZhi6ks-HMqA1PJD27jTak,590
@@ -273,15 +273,15 @@ phoenix/server/static/apple-touch-icon-76x76.png,sha256=CT_xT12I0u2i0WU8JzBZBuOQ
273
273
  phoenix/server/static/apple-touch-icon.png,sha256=fOfpjqGpWYbJ0eAurKsyoZP1EAs6ZVooBJ_SGk2ZkDs,3801
274
274
  phoenix/server/static/favicon.ico,sha256=bY0vvCKRftemZfPShwZtE93DiiQdaYaozkPGwNFr6H8,34494
275
275
  phoenix/server/static/modernizr.js,sha256=mvK-XtkNqjOral-QvzoqsyOMECXIMu5BQwSVN_wcU9c,2564
276
- phoenix/server/static/.vite/manifest.json,sha256=000E_s3Pb1oWXLmvnSBlT88eingszlTB7naM3fEDGEk,1929
277
- phoenix/server/static/assets/components-Csu8UKOs.js,sha256=3ZcTDRJ_RFaer1h-ug94RxeVF6myZpiJo539qA06tGU,299815
278
- phoenix/server/static/assets/index-Bk5C9EA7.js,sha256=e_ymlns5so5gtYryz2HYqDLOmY8Rg8b7ndidAQYjcVQ,7162
279
- phoenix/server/static/assets/pages-UeWaKXNs.js,sha256=A7TKkXM6VXn7OSHyxCf4wdVeeUjVLap8e8n1d7cBdIg,584866
280
- phoenix/server/static/assets/vendor-CtqfhlbC.js,sha256=7uaV_LYzq1C7AchpBdEOl9HpMP1oIpQPYgr8Xj-QtqU,10898284
276
+ phoenix/server/static/.vite/manifest.json,sha256=k6i5VszmBt1XcFCELz8c-7e6TP_x3amoTODdfstyDcA,1929
277
+ phoenix/server/static/assets/components-MllbfxfJ.js,sha256=mkB80JoeM-3PN4jY7reEh8NWoCza0xPP89dwgJ0dizc,304177
278
+ phoenix/server/static/assets/index-BVO2YcT1.js,sha256=ChIu9IaqdcdOJ89wC9wtTPQuN4KGw7HarePwximdaLs,7162
279
+ phoenix/server/static/assets/pages-BHfC6jnL.js,sha256=ci7akTAeDEYkrBXLYwU7AtgTbfBn9rPvORPt0e617YQ,602565
280
+ phoenix/server/static/assets/vendor-BEuNhfwH.js,sha256=im6x5624m4r_P-V7JnxGOr2oX7U9-HOCpXFxL570rA0,10898284
281
281
  phoenix/server/static/assets/vendor-DxkFTwjz.css,sha256=nZrkr0u6NNElFGvpWHk9GTHeGoibCXCli1bE7mXZGZg,1816
282
- phoenix/server/static/assets/vendor-arizeai-C_3SBz56.js,sha256=SF3TtCW_a0L_ISOpA_bqE_1P13tr3gc2LwJ8rvaMtwI,307006
283
- phoenix/server/static/assets/vendor-codemirror-wfdk9cjp.js,sha256=GxQzArx-3Haec8ink-YNyVRPg8-jcQ07AGxGvlloW0Y,392705
284
- phoenix/server/static/assets/vendor-recharts-BiVnSv90.js,sha256=pHVVRhBwXcbgYKImvRLOaLjSgk9woIAId_HvAIMC5D4,282859
282
+ phoenix/server/static/assets/vendor-arizeai-Bskhzyjm.js,sha256=afaoNHydOIeqYuzh_uvsyA-yRVOr6sheR8hyKq7cUlA,307006
283
+ phoenix/server/static/assets/vendor-codemirror-DLlXCf0x.js,sha256=myPIgiviHBKD70ohoYZnBRoy5a7UDyICuyJOqBt7rHc,392705
284
+ phoenix/server/static/assets/vendor-recharts-CRqhvLYg.js,sha256=mX6uPoAPxX1q9_1rJWjmDv1mpk6qgCkd15hzJKfT0Qs,282859
285
285
  phoenix/server/static/assets/vendor-three-DwGkEfCM.js,sha256=0D12ZgKzfKCTSdSTKJBFR2RZO_xxeMXrqDp0AszZqHY,620972
286
286
  phoenix/server/templates/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
287
287
  phoenix/server/templates/index.html,sha256=ram6sfy2obf_F053ay35V30v-mnRWZ86rK-PstXLy1c,4457
@@ -322,9 +322,9 @@ phoenix/utilities/project.py,sha256=auVpARXkDb-JgeX5f2aStyFIkeKvGwN9l7qrFeJMVxI,
322
322
  phoenix/utilities/re.py,sha256=x8Xbk-Wa6qDMAtUd_7JtZvKtrYEuMY-bchB0n163_5c,2006
323
323
  phoenix/utilities/span_store.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
324
324
  phoenix/utilities/template_formatters.py,sha256=JuOyvukMPLDHa1uVNw0kCFBUnIxy02dwAWNZimdIZU4,2423
325
- arize_phoenix-5.7.0.dist-info/METADATA,sha256=_G0Cy0Pqe-U67j83tXVj-RLIbf2czZb8qA1riNKYUeM,22659
326
- arize_phoenix-5.7.0.dist-info/WHEEL,sha256=1yFddiXMmvYK7QYTqtRNtX66WJ0Mz8PYEiEUoOUUxRY,87
327
- arize_phoenix-5.7.0.dist-info/entry_points.txt,sha256=Pgpn8Upxx9P8z8joPXZWl2LlnAlGc3gcQoVchb06X1Q,94
328
- arize_phoenix-5.7.0.dist-info/licenses/IP_NOTICE,sha256=JBqyyCYYxGDfzQ0TtsQgjts41IJoa-hiwDrBjCb9gHM,469
329
- arize_phoenix-5.7.0.dist-info/licenses/LICENSE,sha256=HFkW9REuMOkvKRACuwLPT0hRydHb3zNg-fdFt94td18,3794
330
- arize_phoenix-5.7.0.dist-info/RECORD,,
325
+ arize_phoenix-5.8.0.dist-info/METADATA,sha256=VALVEFvoQcr28Xiu2cMyzl2a17SLOQoFEkiTmkMnkcs,22613
326
+ arize_phoenix-5.8.0.dist-info/WHEEL,sha256=3U_NnUcV_1B1kPkYaPzN-irRckL5VW_lytn0ytO_kRY,87
327
+ arize_phoenix-5.8.0.dist-info/entry_points.txt,sha256=Pgpn8Upxx9P8z8joPXZWl2LlnAlGc3gcQoVchb06X1Q,94
328
+ arize_phoenix-5.8.0.dist-info/licenses/IP_NOTICE,sha256=JBqyyCYYxGDfzQ0TtsQgjts41IJoa-hiwDrBjCb9gHM,469
329
+ arize_phoenix-5.8.0.dist-info/licenses/LICENSE,sha256=HFkW9REuMOkvKRACuwLPT0hRydHb3zNg-fdFt94td18,3794
330
+ arize_phoenix-5.8.0.dist-info/RECORD,,
@@ -1,4 +1,4 @@
1
1
  Wheel-Version: 1.0
2
- Generator: hatchling 1.25.0
2
+ Generator: hatchling 1.26.1
3
3
  Root-Is-Purelib: true
4
4
  Tag: py3-none-any
phoenix/config.py CHANGED
@@ -146,9 +146,10 @@ ENV_PHOENIX_SMTP_VALIDATE_CERTS = "PHOENIX_SMTP_VALIDATE_CERTS"
146
146
  Whether to validate SMTP server certificates. Defaults to true.
147
147
  """
148
148
 
149
- # FastAPI and GQL extension settings
149
+ # API extension settings
150
150
  ENV_PHOENIX_FASTAPI_MIDDLEWARE_PATHS = "PHOENIX_FASTAPI_MIDDLEWARE_PATHS"
151
151
  ENV_PHOENIX_GQL_EXTENSION_PATHS = "PHOENIX_GQL_EXTENSION_PATHS"
152
+ ENV_PHOENIX_GRPC_INTERCEPTOR_PATHS = "PHOENIX_GRPC_INTERCEPTOR_PATHS"
152
153
 
153
154
 
154
155
  def server_instrumentation_is_enabled() -> bool:
@@ -656,7 +657,7 @@ def get_env_db_logging_level() -> int:
656
657
 
657
658
 
658
659
  def get_env_fastapi_middleware_paths() -> list[tuple[str, str]]:
659
- env_value = os.getenv("PHOENIX_FASTAPI_MIDDLEWARE_PATHS", "")
660
+ env_value = os.getenv(ENV_PHOENIX_FASTAPI_MIDDLEWARE_PATHS, "")
660
661
  paths = []
661
662
  for entry in env_value.split(","):
662
663
  entry = entry.strip()
@@ -671,7 +672,7 @@ def get_env_fastapi_middleware_paths() -> list[tuple[str, str]]:
671
672
 
672
673
 
673
674
  def get_env_gql_extension_paths() -> list[tuple[str, str]]:
674
- env_value = os.getenv("PHOENIX_GQL_EXTENSION_PATHS", "")
675
+ env_value = os.getenv(ENV_PHOENIX_GQL_EXTENSION_PATHS, "")
675
676
  paths = []
676
677
  for entry in env_value.split(","):
677
678
  entry = entry.strip()
@@ -685,6 +686,21 @@ def get_env_gql_extension_paths() -> list[tuple[str, str]]:
685
686
  return paths
686
687
 
687
688
 
689
+ def get_env_grpc_interceptor_paths() -> list[tuple[str, str]]:
690
+ env_value = os.getenv(ENV_PHOENIX_GRPC_INTERCEPTOR_PATHS, "")
691
+ paths = []
692
+ for entry in env_value.split(","):
693
+ entry = entry.strip()
694
+ if entry:
695
+ if ":" not in entry:
696
+ raise ValueError(
697
+ f"Invalid interceptor entry '{entry}'. Expected format 'file_path:ClassName'."
698
+ )
699
+ file_path, object_name = entry.split(":", 1)
700
+ paths.append((file_path.strip(), object_name.strip()))
701
+ return paths
702
+
703
+
688
704
  def _get_logging_level(env_var: str, default_level: int) -> int:
689
705
  logging_level = os.getenv(env_var)
690
706
  if not logging_level:
@@ -1,13 +1,12 @@
1
+ import asyncio
1
2
  import importlib.util
3
+ import inspect
4
+ import json
5
+ import time
2
6
  from abc import ABC, abstractmethod
3
7
  from collections.abc import AsyncIterator, Callable, Iterator
4
- from typing import (
5
- TYPE_CHECKING,
6
- Any,
7
- Mapping,
8
- Optional,
9
- Union,
10
- )
8
+ from functools import wraps
9
+ from typing import TYPE_CHECKING, Any, Hashable, Mapping, Optional, Union
11
10
 
12
11
  from openinference.instrumentation import safe_json_dumps
13
12
  from openinference.semconv.trace import SpanAttributes
@@ -15,10 +14,14 @@ from strawberry import UNSET
15
14
  from strawberry.scalars import JSON as JSONScalarType
16
15
  from typing_extensions import TypeAlias, assert_never
17
16
 
18
- from phoenix.server.api.helpers.playground_registry import (
19
- PROVIDER_DEFAULT,
20
- register_llm_client,
17
+ from phoenix.evals.models.rate_limiters import (
18
+ AsyncCallable,
19
+ GenericType,
20
+ ParameterSpec,
21
+ RateLimiter,
22
+ RateLimitError,
21
23
  )
24
+ from phoenix.server.api.helpers.playground_registry import PROVIDER_DEFAULT, register_llm_client
22
25
  from phoenix.server.api.input_types.GenerativeModelInput import GenerativeModelInput
23
26
  from phoenix.server.api.input_types.InvocationParameters import (
24
27
  BoundedFloatInvocationParameter,
@@ -42,16 +45,99 @@ from phoenix.server.api.types.GenerativeProvider import GenerativeProviderKey
42
45
  if TYPE_CHECKING:
43
46
  from anthropic.types import MessageParam
44
47
  from openai.types import CompletionUsage
45
- from openai.types.chat import (
46
- ChatCompletionMessageParam,
47
- ChatCompletionMessageToolCallParam,
48
- )
48
+ from openai.types.chat import ChatCompletionMessageParam, ChatCompletionMessageToolCallParam
49
49
 
50
50
  DependencyName: TypeAlias = str
51
51
  SetSpanAttributesFn: TypeAlias = Callable[[Mapping[str, Any]], None]
52
52
  ChatCompletionChunk: TypeAlias = Union[TextChunk, ToolCallChunk]
53
53
 
54
54
 
55
+ class KeyedSingleton:
56
+ _instances: dict[Hashable, "KeyedSingleton"] = {}
57
+
58
+ def __new__(cls, *args: Any, **kwargs: Any) -> "KeyedSingleton":
59
+ if "singleton_key" in kwargs:
60
+ singleton_key = kwargs.pop("singleton_key")
61
+ elif args:
62
+ singleton_key = args[0]
63
+ args = args[1:]
64
+ else:
65
+ raise ValueError("singleton_key must be provided")
66
+
67
+ instance_key = (cls, singleton_key)
68
+ if instance_key not in cls._instances:
69
+ instance = super().__new__(cls)
70
+ cls._instances[instance_key] = instance
71
+ return cls._instances[instance_key]
72
+
73
+
74
+ class PlaygroundRateLimiter(RateLimiter, KeyedSingleton):
75
+ """
76
+ A rate rate limiter class that will be instantiated once per `singleton_key`.
77
+ """
78
+
79
+ def __init__(self, singleton_key: Hashable, rate_limit_error: Optional[type[BaseException]]):
80
+ super().__init__(
81
+ rate_limit_error=rate_limit_error,
82
+ max_rate_limit_retries=3,
83
+ initial_per_second_request_rate=2.0,
84
+ maximum_per_second_request_rate=10.0,
85
+ enforcement_window_minutes=1,
86
+ rate_reduction_factor=0.5,
87
+ rate_increase_factor=0.01,
88
+ cooldown_seconds=5,
89
+ verbose=False,
90
+ )
91
+
92
+ # TODO: update the rate limiter class in phoenix.evals to support decorated sync functions
93
+ def _alimit(
94
+ self, fn: Callable[ParameterSpec, GenericType]
95
+ ) -> AsyncCallable[ParameterSpec, GenericType]:
96
+ @wraps(fn)
97
+ async def wrapper(*args: Any, **kwargs: Any) -> GenericType:
98
+ self._initialize_async_primitives()
99
+ assert self._rate_limit_handling_lock is not None and isinstance(
100
+ self._rate_limit_handling_lock, asyncio.Lock
101
+ )
102
+ assert self._rate_limit_handling is not None and isinstance(
103
+ self._rate_limit_handling, asyncio.Event
104
+ )
105
+ try:
106
+ try:
107
+ await asyncio.wait_for(self._rate_limit_handling.wait(), 120)
108
+ except asyncio.TimeoutError:
109
+ self._rate_limit_handling.set() # Set the event as a failsafe
110
+ await self._throttler.async_wait_until_ready()
111
+ request_start_time = time.time()
112
+ if inspect.iscoroutinefunction(fn):
113
+ return await fn(*args, **kwargs) # type: ignore
114
+ else:
115
+ return fn(*args, **kwargs)
116
+ except self._rate_limit_error:
117
+ async with self._rate_limit_handling_lock:
118
+ self._rate_limit_handling.clear() # prevent new requests from starting
119
+ self._throttler.on_rate_limit_error(request_start_time, verbose=self._verbose)
120
+ try:
121
+ for _attempt in range(self._max_rate_limit_retries):
122
+ try:
123
+ request_start_time = time.time()
124
+ await self._throttler.async_wait_until_ready()
125
+ if inspect.iscoroutinefunction(fn):
126
+ return await fn(*args, **kwargs) # type: ignore
127
+ else:
128
+ return fn(*args, **kwargs)
129
+ except self._rate_limit_error:
130
+ self._throttler.on_rate_limit_error(
131
+ request_start_time, verbose=self._verbose
132
+ )
133
+ continue
134
+ finally:
135
+ self._rate_limit_handling.set() # allow new requests to start
136
+ raise RateLimitError(f"Exceeded max ({self._max_rate_limit_retries}) retries")
137
+
138
+ return wrapper
139
+
140
+
55
141
  class PlaygroundStreamingClient(ABC):
56
142
  def __init__(
57
143
  self,
@@ -150,10 +236,12 @@ class OpenAIStreamingClient(PlaygroundStreamingClient):
150
236
  api_key: Optional[str] = None,
151
237
  ) -> None:
152
238
  from openai import AsyncOpenAI
239
+ from openai import RateLimitError as OpenAIRateLimitError
153
240
 
154
241
  super().__init__(model=model, api_key=api_key)
155
242
  self.client = AsyncOpenAI(api_key=api_key)
156
243
  self.model_name = model.name
244
+ self.rate_limiter = PlaygroundRateLimiter(model.provider_key, OpenAIRateLimitError)
157
245
 
158
246
  @classmethod
159
247
  def dependencies(cls) -> list[DependencyName]:
@@ -174,19 +262,16 @@ class OpenAIStreamingClient(PlaygroundStreamingClient):
174
262
  invocation_name="max_tokens",
175
263
  canonical_name=CanonicalParameterName.MAX_COMPLETION_TOKENS,
176
264
  label="Max Tokens",
177
- default_value=UNSET,
178
265
  ),
179
266
  BoundedFloatInvocationParameter(
180
267
  invocation_name="frequency_penalty",
181
268
  label="Frequency Penalty",
182
- default_value=UNSET,
183
269
  min_value=-2.0,
184
270
  max_value=2.0,
185
271
  ),
186
272
  BoundedFloatInvocationParameter(
187
273
  invocation_name="presence_penalty",
188
274
  label="Presence Penalty",
189
- default_value=UNSET,
190
275
  min_value=-2.0,
191
276
  max_value=2.0,
192
277
  ),
@@ -194,13 +279,11 @@ class OpenAIStreamingClient(PlaygroundStreamingClient):
194
279
  invocation_name="stop",
195
280
  canonical_name=CanonicalParameterName.STOP_SEQUENCES,
196
281
  label="Stop Sequences",
197
- default_value=UNSET,
198
282
  ),
199
283
  BoundedFloatInvocationParameter(
200
284
  invocation_name="top_p",
201
285
  canonical_name=CanonicalParameterName.TOP_P,
202
286
  label="Top P",
203
- default_value=UNSET,
204
287
  min_value=0.0,
205
288
  max_value=1.0,
206
289
  ),
@@ -208,20 +291,16 @@ class OpenAIStreamingClient(PlaygroundStreamingClient):
208
291
  invocation_name="seed",
209
292
  canonical_name=CanonicalParameterName.RANDOM_SEED,
210
293
  label="Seed",
211
- default_value=UNSET,
212
294
  ),
213
295
  JSONInvocationParameter(
214
296
  invocation_name="tool_choice",
215
297
  label="Tool Choice",
216
298
  canonical_name=CanonicalParameterName.TOOL_CHOICE,
217
- default_value=UNSET,
218
- hidden=True,
219
299
  ),
220
300
  JSONInvocationParameter(
221
301
  invocation_name="response_format",
222
302
  label="Response Format",
223
303
  canonical_name=CanonicalParameterName.RESPONSE_FORMAT,
224
- default_value=UNSET,
225
304
  ),
226
305
  ]
227
306
 
@@ -240,7 +319,8 @@ class OpenAIStreamingClient(PlaygroundStreamingClient):
240
319
  openai_messages = [self.to_openai_chat_completion_param(*message) for message in messages]
241
320
  tool_call_ids: dict[int, str] = {}
242
321
  token_usage: Optional["CompletionUsage"] = None
243
- async for chunk in await self.client.chat.completions.create(
322
+ throttled_create = self.rate_limiter.alimit(self.client.chat.completions.create)
323
+ async for chunk in await throttled_create(
244
324
  messages=openai_messages,
245
325
  model=self.model_name,
246
326
  stream=True,
@@ -370,20 +450,16 @@ class OpenAIO1StreamingClient(OpenAIStreamingClient):
370
450
  invocation_name="max_completion_tokens",
371
451
  canonical_name=CanonicalParameterName.MAX_COMPLETION_TOKENS,
372
452
  label="Max Completion Tokens",
373
- default_value=UNSET,
374
453
  ),
375
454
  IntInvocationParameter(
376
455
  invocation_name="seed",
377
456
  canonical_name=CanonicalParameterName.RANDOM_SEED,
378
457
  label="Seed",
379
- default_value=UNSET,
380
458
  ),
381
459
  JSONInvocationParameter(
382
460
  invocation_name="tool_choice",
383
461
  label="Tool Choice",
384
462
  canonical_name=CanonicalParameterName.TOOL_CHOICE,
385
- default_value=UNSET,
386
- hidden=True,
387
463
  ),
388
464
  ]
389
465
 
@@ -409,7 +485,8 @@ class OpenAIO1StreamingClient(OpenAIStreamingClient):
409
485
 
410
486
  tool_call_ids: dict[int, str] = {}
411
487
 
412
- response = await self.client.chat.completions.create(
488
+ throttled_create = self.rate_limiter.alimit(self.client.chat.completions.create)
489
+ response = await throttled_create(
413
490
  messages=openai_messages,
414
491
  model=self.model_name,
415
492
  tools=tools or NOT_GIVEN,
@@ -544,6 +621,7 @@ class AnthropicStreamingClient(PlaygroundStreamingClient):
544
621
  super().__init__(model=model, api_key=api_key)
545
622
  self.client = anthropic.AsyncAnthropic(api_key=api_key)
546
623
  self.model_name = model.name
624
+ self.rate_limiter = PlaygroundRateLimiter(model.provider_key, anthropic.RateLimitError)
547
625
 
548
626
  @classmethod
549
627
  def dependencies(cls) -> list[DependencyName]:
@@ -556,14 +634,12 @@ class AnthropicStreamingClient(PlaygroundStreamingClient):
556
634
  invocation_name="max_tokens",
557
635
  canonical_name=CanonicalParameterName.MAX_COMPLETION_TOKENS,
558
636
  label="Max Tokens",
559
- default_value=UNSET,
560
637
  required=True,
561
638
  ),
562
639
  BoundedFloatInvocationParameter(
563
640
  invocation_name="temperature",
564
641
  canonical_name=CanonicalParameterName.TEMPERATURE,
565
642
  label="Temperature",
566
- default_value=UNSET,
567
643
  min_value=0.0,
568
644
  max_value=1.0,
569
645
  ),
@@ -571,13 +647,11 @@ class AnthropicStreamingClient(PlaygroundStreamingClient):
571
647
  invocation_name="stop_sequences",
572
648
  canonical_name=CanonicalParameterName.STOP_SEQUENCES,
573
649
  label="Stop Sequences",
574
- default_value=UNSET,
575
650
  ),
576
651
  BoundedFloatInvocationParameter(
577
652
  invocation_name="top_p",
578
653
  canonical_name=CanonicalParameterName.TOP_P,
579
654
  label="Top P",
580
- default_value=UNSET,
581
655
  min_value=0.0,
582
656
  max_value=1.0,
583
657
  ),
@@ -585,8 +659,6 @@ class AnthropicStreamingClient(PlaygroundStreamingClient):
585
659
  invocation_name="tool_choice",
586
660
  label="Tool Choice",
587
661
  canonical_name=CanonicalParameterName.TOOL_CHOICE,
588
- default_value=UNSET,
589
- hidden=True,
590
662
  ),
591
663
  ]
592
664
 
@@ -608,9 +680,11 @@ class AnthropicStreamingClient(PlaygroundStreamingClient):
608
680
  "model": self.model_name,
609
681
  "system": system_prompt,
610
682
  "max_tokens": 1024,
683
+ "tools": tools,
611
684
  **invocation_parameters,
612
685
  }
613
- async with self.client.messages.stream(**anthropic_params) as stream:
686
+ throttled_stream = self.rate_limiter._alimit(self.client.messages.stream)
687
+ async with await throttled_stream(**anthropic_params) as stream:
614
688
  async for event in stream:
615
689
  if isinstance(event, anthropic_types.RawMessageStartEvent):
616
690
  self._attributes.update(
@@ -622,6 +696,18 @@ class AnthropicStreamingClient(PlaygroundStreamingClient):
622
696
  self._attributes.update(
623
697
  {LLM_TOKEN_COUNT_COMPLETION: event.message.usage.output_tokens}
624
698
  )
699
+ elif (
700
+ isinstance(event, anthropic_streaming.ContentBlockStopEvent)
701
+ and event.content_block.type == "tool_use"
702
+ ):
703
+ tool_call_chunk = ToolCallChunk(
704
+ id=event.content_block.id,
705
+ function=FunctionCallChunk(
706
+ name=event.content_block.name,
707
+ arguments=json.dumps(event.content_block.input),
708
+ ),
709
+ )
710
+ yield tool_call_chunk
625
711
  elif isinstance(
626
712
  event,
627
713
  (
@@ -629,6 +715,7 @@ class AnthropicStreamingClient(PlaygroundStreamingClient):
629
715
  anthropic_types.RawContentBlockDeltaEvent,
630
716
  anthropic_types.RawMessageDeltaEvent,
631
717
  anthropic_streaming.ContentBlockStopEvent,
718
+ anthropic_streaming.InputJsonEvent,
632
719
  ),
633
720
  ):
634
721
  # event types emitted by the stream that don't contain useful information