A Python client library for the AI DIAL API
Project description
AI DIAL Client (Python)
- Usage
- Development
Usage
This section outlines how to use the AI DIAL Python client to interact with the DIAL Core API. It covers authentication methods, making chat completion requests, working with files, managing applications, and utilizing client pools for efficient connection management.
Authentication
API Keys
For authentication with an API key, pass it during the client initialization:
from aidial_client import Dial, AsyncDial
dial_client = Dial(api_key="your_api_key", base_url="https://your-dial-instance.com")
async_dial_client = AsyncDial(
api_key="your_api_key", base_url="https://your-dial-instance.com"
)
You can also pass api_key as a function without parameters, that returns a string:
def my_key_function():
# Any custom logic to get an API key
return "your-api-key"
dial_client = Dial(api_key=my_key_function, base_url="https://your-dial-instance.com")
async_dial_client = AsyncDial(
api_key=my_key_function, base_url="https://your-dial-instance.com"
)
For async clients, you can use coroutine as well:
async def my_key_function():
# Any custom logic to get an API key
return "your-api-key"
async_dial_client = AsyncDial(
api_key=my_key_function, base_url="https://your-dial-instance.com"
)
Bearer Token
You can use a Bearer Token for a token-based authentication of API calls. Client instances will use it to construct the Authorization header when making requests:
from aidial_client import Dial, AsyncDial
# Create an instance of the synchronous client
sync_client = Dial(
bearer_token="your_bearer_token_here", base_url="https://your-dial-instance.com"
)
# Create an instance of the asynchronous client
async_client = AsyncDial(
bearer_token="your_bearer_token_here", base_url="https://your-dial-instance.com"
)
You can also pass bearer_token as a function without parameters, that returns a string:
def my_token_function():
# Any custom logic to get an API key
return "your-bearer-token"
dial_client = Dial(
bearer_token=my_token_function, base_url="https://your-dial-instance.com"
)
async_dial_client = AsyncDial(
bearer_token=my_token_function, base_url="https://your-dial-instance.com"
)
For async clients, you can use coroutine as well:
async def my_token_function():
# Any custom logic to get a bearer token
return "your-bearer-token"
dial_client = Dial(
bearer_token=my_token_function, base_url="https://your-dial-instance.com"
)
Deployments
List Deployments
To get a list of available deployments:
# Sync
deployments = client.deployments.list()
# Async
deployments = await async_client.deployments.list()
>>> client.deployments.list()
[
Deployment(id='gpt-35-turbo', model='gpt-35-turbo', owner='organization-owner', object='deployment', status='succeeded', created_at=1724760524, updated_at=1724760524, scale_settings=ScaleSettings(scale_type='standard'), features=Features(rate=False, tokenize=False, truncate_prompt=False, configuration=False, system_prompt=True, tools=False, seed=False, url_attachments=False, folder_attachments=False, allow_resume=True)),
Deployment(id='stable-diffusion-xl', model='stable-diffusion-xl', owner='organization-owner', object='deployment', status='succeeded', created_at=1724760524, updated_at=1724760524, scale_settings=ScaleSettings(scale_type='standard'), features=Features(rate=False, tokenize=False, truncate_prompt=False, configuration=False, system_prompt=True, tools=False, seed=False, url_attachments=False, folder_attachments=False, allow_resume=True)),
...,
]
Get Deployment by Id
To fetch a single deployment by its identifier:
# Sync
deployment = client.deployments.get("gpt-35-turbo")
# Async
deployment = await async_client.deployments.get("gpt-35-turbo")
As a result, you will receive a Deployment object:
Deployment(
id="gpt-35-turbo",
model="gpt-35-turbo",
object="deployment",
owner="organization-owner",
status="succeeded",
created_at=1724760524,
updated_at=1724760524,
scale_settings=ScaleSettings(scale_type="standard"),
features=Features(
rate=False,
tokenize=False,
truncate_prompt=False,
configuration=True,
system_prompt=True,
tools=True,
seed=False,
url_attachments=False,
folder_attachments=False,
allow_resume=True,
),
defaults={},
)
Get Deployment Configuration
Some deployments expose a JSON Schema document describing their runtime configuration. Use get_configuration() to retrieve it:
# Sync
config = client.deployments.get_configuration_schema("gpt-35-turbo")
# Async
config = await async_client.deployments.get_configuration_schema("gpt-35-turbo")
The response is a plain dict whose shape is entirely deployment-specific:
{
"type": "object",
"properties": {
"model_to_use": {
"type": "string",
"enum": ["gpt-4", "gpt-4o"],
"default": "gpt-4",
}
},
"additionalProperties": False,
}
Make Completions Requests
Without Streaming
Synchronous:
...
client = Dial(api_key="your-api-key", base_url="https://your-dial-instance.com")
completion = client.chat.completions.create(
deployment_name="gpt-35-turbo",
stream=False,
messages=[
{
"role": "system",
"content": "2+3=",
}
],
api_version="2024-02-15-preview",
)
Asynchronous:
...
async_client = AsyncDial(
api_key="your-api-key", base_url="https://your-dial-instance.com"
)
completion = await async_client.chat.completions.create(
deployment_name="gpt-35-turbo",
stream=False,
messages=[
{
"role": "system",
"content": "2+3=",
}
],
api_version="2024-02-15-preview",
)
Example of a response:
>>> completion
ChatCompletionResponse(
id='chatcmpl-A18H6rWmocm52WMweXvp8BNnwbfsp',
object='chat.completion',
choices=[
Choice(
index=0,
message=ChatCompletionMessage(
role='assistant',
content='5',
custom_content=None,
function_call=None,
tool_calls=None
),
finish_reason='stop',
logprobs=None
)
],
created=1724833500,
model='gpt-35-turbo-16k',
usage=CompletionUsage(
prompt_tokens=11,
completion_tokens=1,
total_tokens=12
),
system_fingerprint=None
)
With Streaming
Synchronous:
...
client = Dial(api_key="your-api-key", base_url="https://your-dial-instance.com")
completion = client.chat.completions.create(
deployment_name="gpt-35-turbo",
# Specify a stream parameter
stream=True,
messages=[
{
"role": "system",
"content": "2+3=",
}
],
api_version="2024-02-15-preview",
)
for chunk in completion:
...
Asynchronous:
...
async_client = AsyncDial(
api_key="your-api-key", base_url="https://your-dial-instance.com"
)
completion = await async_client.chat.completions.create(
deployment_name="gpt-35-turbo",
# Specify a stream parameter
stream=True,
messages=[
{
"role": "system",
"content": "2+3=",
}
],
api_version="2024-02-15-preview",
)
async for chunk in completion:
...
Example of chunk objects:
>>> chunk
ChatCompletionChunk(
id='chatcmpl-A18NiK8Zh39RdcNX91T0eHfERfyU3',
object='chat.completion.chunk',
choices=[
ChoiceDelta(
index=0,
delta=ChunkEmptyDelta(
content='5',
object=None,
tool_calls=None,
role=None
),
finish_reason=None,
logprobs=None
)
],
created=1724833910,
model='gpt-35-turbo-16k',
usage=None,
system_fingerprint=None
)
>>> chunk
ChatCompletionChunk(
id='chatcmpl-A18NiK8Zh39RdcNX91T0eHfERfyU3',
object='chat.completion.chunk',
choices=[
ChoiceDelta(
index=0,
delta=ChunkEmptyDelta(
content=None,
object=None,
tool_calls=None,
role=None
),
# Last chunk has non-empty finish_reason
finish_reason='stop',
logprobs=None
)
],
created=1724833910,
model='gpt-35-turbo-16k',
usage=CompletionUsage(
prompt_tokens=11,
completion_tokens=1,
total_tokens=12
),
system_fingerprint=None
)
Working with Files
Working with URLs
Files are AI DIAL resources that operate with URL-like objects. Use pathlib.PurePosixPath or str to create to create new URL-like objects or to get a string representation of them.
- Use
client.my_files_home()to upload a file into your bucket in the AI DIAL storage. - Use
await async_client.my_files_home()to get the URL of your bucket and then use it to upload files.
The following example demonstrates how you can use the path-like object returned by my_files_home() function:
sync_client.files.upload(
url=sync_client.my_files_home() / "some-relative-path/my-file.txt", ...
)
async_client.files.upload(
url=await async_client.my_files_home() / "some-relative-path/my-file.txt", ...
)
If you already have a relative URL like files/..., you can use it as well:
relative_url = "files/test-bucket/some-relative-path/my-file.txt"
sync_client.files.upload(url=relative_url, ...)
You can also use an absolute URL:
absolute_url = "http://dial.core/v1/files/test-bucket/some-relative-path/my-file.txt"
sync_client.files.upload(url=absolute_url, ...)
Note, that an invalid URL provided to the function, will raise an InvalidDialURLException exception.
Uploading Files
Use upload() to add files into your storage bucket:
with open("./some-local-file.txt", "rb") as file:
# Sync client
sync_client.files.upload(
url=sync_client.my_files_home() / "some-relative-path/my-file.txt", file=file
)
# Async client
await async_client.files.upload(
url=await async_client.my_files_home() / "some-relative-path/my-file.txt",
file=file,
)
Files can contain raw bytes or file-like objects. To specify filename and content type of the uploaded file, use tuple instead of file object:
sync_client.files.upload(
url=sync_client.my_files_home() / "some-relative-path/my-file.txt",
file=("filename.txt", "text/plain", file),
)
Downloading Files
Use download() to download files from your storage bucket:
result = client.files.download(
url=client.my_files_home() / "relative_folder/my-file.txt"
)
result = await async_client.files.download(
url=await async_client.my_files_home() / "relative_folder/my-file.txt"
)
As a result, you will receive an object of type FileDownloadResponse, that you can iterate by byte chunks:
for bytes_chunk in result:
...
or get full content as bytes:
# Sync
all_content = result.get_content()
# Async
all_content = await result.aget_content()
or write it to the file:
# Sync
result.write_to("./some-local-file.txt")
# Async
await result.awrite_to("./some-local-file.txt")
Deleting Files
Use delete() to remove files from your storage bucket:
await sync_client.files.delete(
url=sync_client.my_files_home() / "relative_folder/my-file.txt"
)
await async_client.files.delete(
url=await async_client.my_files_home() / "relative_folder/my-file.txt"
)
Accessing Metadata
Use metadata() to access metadata of a file:
metadata = await async_client.files.metadata(
url=await async_client.my_files_home() / "relative_folder/my-file.txt"
)
Example of metadata:
FileMetadata(
name="my-file.txt",
parent_path="relative_folder",
bucket="my-bucket",
url="files/my-bucket/test-folder-artifacts/test-file",
node_type="ITEM",
resource_type="FILE",
content_length=12,
content_type="application/octet-stream",
items=None,
updatedAt=1724836248936,
etag="9749fad13d6e7092a6337c4af9d83764",
createdAt=1724836229736,
)
Prompts
Get Prompt
Use get() to fetch a single prompt by its storage path:
# Sync
prompt = client.prompts.get("prompts/my-bucket/my-folder/my-prompt")
# Async
prompt = await async_client.prompts.get("prompts/my-bucket/my-folder/my-prompt")
As a result, you will receive a Prompt object:
Prompt(
id="prompts/my-bucket/my-folder/my-prompt",
name="my-prompt",
folder_id="my-folder",
content="You are a helpful assistant.",
)
Get Prompt Metadata
Use get_metadata() to access metadata of a prompt:
# Sync
metadata = client.prompts.get_metadata("prompts/my-bucket/my-folder/my-prompt")
# Async
metadata = await async_client.prompts.get_metadata(
"prompts/my-bucket/my-folder/my-prompt"
)
As a result, you will receive a PromptMetadata object:
PromptMetadata(
name="my-prompt",
parent_path="my-folder",
bucket="my-bucket",
url="prompts/my-bucket/my-folder/my-prompt",
node_type="ITEM",
resource_type="PROMPT",
items=[],
)
Applications
List Applications
To get a list of your DIAL applications:
# Sync
applications = client.application.list()
# Async
applications = await async_client.application.list()
As a result, you will receive a list of Application objects:
[
Application(
object="application",
id="app_id",
description="",
application="app_id",
display_name="app with attachments",
display_version="0.0.0",
icon_url="...",
reference="...",
owner="organization-owner",
status="succeeded",
created_at=1672534800,
updated_at=1672534800,
features=Features(
rate=False,
tokenize=False,
truncate_prompt=False,
configuration=False,
system_prompt=True,
tools=False,
seed=False,
url_attachments=False,
folder_attachments=False,
allow_resume=True,
),
input_attachment_types=["image/png", "text/txt", "image/jpeg"],
defaults={},
max_input_attachments=0,
description_keywords=[],
),
...,
]
Get Application by Id
You can get your DIAL applications by their Ids:
# Sync
application = client.application.get("app_id")
# Async
application = await async_client.application.get("app_id")
As a result, you will receive a list of Application objects. Refer to the previous example.
Models
Get Model by Name
To retrieve metadata, capabilities, and pricing for a specific model:
# Sync
model_info = client.model.get("gpt-4")
# Async
model_info = await async_client.model.get("gpt-4")
As a result, you will receive a ModelInfo object:
ModelInfo(
id="gpt-4",
model="gpt-4",
object="model",
owner="organization-owner",
status="succeeded",
created_at=1724760524,
updated_at=1724760524,
lifecycle_status="generally-available",
display_name="GPT-4",
description="OpenAI GPT-4 model.",
capabilities=ModelCapabilities(
scale_types=["standard"],
completion=False,
chat_completion=True,
embeddings=False,
fine_tune=False,
inference=False,
),
limits=ModelLimits(
max_prompt_tokens=8192,
max_completion_tokens=4096,
max_total_tokens=None,
),
pricing=ModelPricing(
unit="token",
prompt="0.00003",
completion="0.00006",
),
)
Toolsets
Get Toolset by Id
To retrieve information about a specific MCP toolset:
# Sync
toolset_info = client.toolset.get("my-toolset")
# Async
toolset_info = await async_client.toolset.get("my-toolset")
As a result, you will receive a ToolsetInfo object:
ToolsetInfo(
id="my-toolset",
toolset="my-toolset",
display_name="My Toolset",
description="A collection of tools for data processing.",
transport="HTTP",
allowed_tools=["tool-a", "tool-b"],
owner="organization-owner",
status="succeeded",
created_at=1724760524,
updated_at=1724760524,
)
Resource Permissions
Grant Permissions
Use resource_permissions.grant() to grant access to one or more files in DIAL storage to a specific deployment (receiver). This is typically used when a deployment needs to read files on behalf of a user.
# Sync
client.resource_permissions.grant(
resources=["files/my-bucket/report.pdf"],
receiver="my-deployment",
permissions=["READ"],
)
# Async
await async_client.resource_permissions.grant(
resources=["files/my-bucket/report.pdf"],
receiver="my-deployment",
permissions=["READ"],
)
resources— list of DIAL file URL strings to share.receiver— the deployment ID that should receive access.permissions— list of permission strings; defaults to["READ"].
The method returns None on success and raises DialException on HTTP error.
Client Pool
When you need to create multiple DIAL clients and wish to enhance performance by reusing the HTTP connection for the same DIAL instance, consider using synchronous and asynchronous client pools.
Synchronous Client Pool
from aidial_client import DialClientPool
client_pool = DialClientPool()
first_client = client_pool.create_client(
base_url="https://your-dial-instance.com", api_key="your-api-key"
)
second_client = client_pool.create_client(
base_url="https://your-dial-instance.com", bearer_token="your-bearer-token"
)
Asynchronous Client Pool
from dial_client import (
AsyncDialClientPool,
)
client_pool = AsyncDialClientPool()
first_client = client_pool.create_client(
base_url="https://your-dial-instance.com", api_key="your-api-key"
)
second_client = client_pool.create_client(
base_url="https://your-dial-instance.com", bearer_token="your-bearer-token"
)
Development
To set up the development environment and run the project, follow the instructions below.
Pre-requisites
The following tools are required to work with the project:
MakePython 3.10Poetry 2.*. Installation guidance can be found here
Setup
- Create
.envfile in the root of the project. Copy.env.templatefile data to the.envand customize the values if needed. You can customize python and poetry locations. - Create and activate virtual environment
make init_env source .venv/bin/activate
- Install dependencies
make install
Git hooks
You may optionally install Git hooks that will automatically run the linting step on Git push. You only need to do it once for the given repository.
make install_git_hooks
[!IMPORTANT] This command doesn't work if you have already installed Git hooks locally or globally.
Main commands
| Command | Description |
|---|---|
make install |
Install virtual environment and dependencies |
make build |
Build the package |
make clean |
Clean virtual environment and build artifacts |
make install_git_hooks |
Install the git hooks |
make lint |
Run linters |
make format |
Run code formatters |
make test |
Run tests (e.g., make test PYTHON=3.12) |
make integration_test |
Run integration tests |
make coverage |
Generate test coverage report |
make help |
Show available commands |
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file aidial_client-0.7.2.tar.gz.
File metadata
- Download URL: aidial_client-0.7.2.tar.gz
- Upload date:
- Size: 34.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/2.3.2 CPython/3.10.20 Linux/6.17.0-1010-azure
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1f1f799c01b90a56cad1cc3d2a438de3c14bd518e60baca6dc7c843dc13c9022
|
|
| MD5 |
ac9b174a70c9b1b44225e118f5654486
|
|
| BLAKE2b-256 |
e8f8dd29c97484f5861db271cc06bb38850a642d48e0f86f12caf43e488a39ae
|
File details
Details for the file aidial_client-0.7.2-py3-none-any.whl.
File metadata
- Download URL: aidial_client-0.7.2-py3-none-any.whl
- Upload date:
- Size: 49.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/2.3.2 CPython/3.10.20 Linux/6.17.0-1010-azure
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
620bdd237c8c7d642e730a42ae02dad2e8ea3c2e3772ed6d9c9f99f0c6fc3fea
|
|
| MD5 |
00a75af4e1990e1ddf70e2d2994822ce
|
|
| BLAKE2b-256 |
f38d842be660a672418f8b1c603b0165de6e001a26bc0d56c62d7623abf43ae2
|