Skip to main content
控制平面 API 是 LangSmith 部署 的一部分。通过控制平面 API,您可以以编程方式创建、管理和自动化您的 Agent Server 部署——例如,作为自定义 CI/CD 工作流的一部分。 在侧边栏的 控制平面 API 部分浏览完整的 API 参考,或参考以下端点分组:

主机地址

云数据区域的控制平面主机地址:
美国欧盟
https://api.host.langchain.comhttps://eu.api.host.langchain.com
注意:LangSmith 的自托管部署将拥有自定义的控制平面主机地址。控制平面 API 可通过路径 /api-host 访问。例如,http(s)://<主机地址>/api-host/v2/deployments。更多详情请参阅 自托管使用指南

认证

要使用控制平面 API 进行认证,请将 X-Api-Key 请求头设置为有效的 LangSmith API 密钥,并将 X-Tenant-Id 请求头设置为要操作的有效工作区 ID。 curl 命令示例:
curl --request GET \
  --url http://localhost:8124/v2/deployments \
  --header 'X-Api-Key: LANGSMITH_API_KEY'
  --header 'X-Tenant-Id': WORKSPACE_ID'

版本控制

每个端点路径都带有版本前缀(例如 v1v2)。

快速开始

  1. 调用 POST /v2/deployments 创建新的部署。响应体包含部署 ID (id) 和最新(也是第一个)修订版本的 ID (latest_revision_id)。
  2. 调用 GET /v2/deployments/{deployment_id} 获取部署信息。将 URL 中的 deployment_id 设置为部署 ID (id) 的值。
  3. 通过调用 GET /v2/deployments/{deployment_id}/revisions/{latest_revision_id} 轮询修订版本的 status,直到 status 变为 DEPLOYED
  4. 调用 PATCH /v2/deployments/{deployment_id} 更新部署。

示例代码

以下是示例 Python 代码,演示了如何编排控制平面 API 来创建部署、更新部署和删除部署。
import os
import time

import requests
from dotenv import load_dotenv


load_dotenv()

# 必需的环境变量
CONTROL_PLANE_HOST = os.getenv("CONTROL_PLANE_HOST")
LANGSMITH_API_KEY = os.getenv("LANGSMITH_API_KEY")
WORKSPACE_ID = os.getenv("WORKSPACE_ID")
INTEGRATION_ID = os.getenv("INTEGRATION_ID")
MAX_WAIT_TIME = 1800  # 30 分钟


def get_headers() -> dict:
    """返回控制平面 API 请求的通用请求头。"""
    return {
        "X-Api-Key": LANGSMITH_API_KEY,
        "X-Tenant-Id": WORKSPACE_ID,
    }


def create_deployment() -> str:
    """创建部署。返回部署 ID。"""
    headers = get_headers()
    headers["Content-Type"] = "application/json"

    deployment_name = "my_deployment"

    request_body = {
        "name": deployment_name,
        "source": "github",
        "source_config": {
            "integration_id": INTEGRATION_ID,
            "repo_url": "https://github.com/langchain-ai/langgraph-example",
            "deployment_type": "dev",
            "build_on_push": False,
            "custom_url": None,
            "resource_spec": None,
        },
        "source_revision_config": {
            "repo_ref": "main",
            "langgraph_config_path": "langgraph.json",
            "image_uri": None,
        },
        "secrets": [
            {
                "name": "OPENAI_API_KEY",
                "value": "test_openai_api_key",
            },
            {
                "name": "ANTHROPIC_API_KEY",
                "value": "test_anthropic_api_key",
            },
            {
                "name": "TAVILY_API_KEY",
                "value": "test_tavily_api_key",
            },
        ],
    }

    response = requests.post(
        url=f"{CONTROL_PLANE_HOST}/v2/deployments",
        headers=headers,
        json=request_body,
    )

    if response.status_code != 201:
        raise Exception(f"创建部署失败: {response.text}")

    deployment_id = response.json()["id"]
    print(f"已创建部署 {deployment_name} ({deployment_id})")
    return deployment_id


def get_deployment(deployment_id: str) -> dict:
    """获取部署信息。"""
    response = requests.get(
        url=f"{CONTROL_PLANE_HOST}/v2/deployments/{deployment_id}",
        headers=get_headers(),
    )

    if response.status_code != 200:
        raise Exception(f"获取部署 ID {deployment_id} 失败: {response.text}")

    return response.json()


def list_revisions(deployment_id: str) -> list[dict]:
    """列出修订版本。

    返回的列表按 created_at 降序排序(最新的在前)。
    """
    response = requests.get(
        url=f"{CONTROL_PLANE_HOST}/v2/deployments/{deployment_id}/revisions",
        headers=get_headers(),
    )

    if response.status_code != 200:
        raise Exception(
            f"列出部署 ID {deployment_id} 的修订版本失败: {response.text}"
        )

    return response.json()


def get_revision(
    deployment_id: str,
    revision_id: str,
) -> dict:
    """获取修订版本信息。"""
    response = requests.get(
        url=f"{CONTROL_PLANE_HOST}/v2/deployments/{deployment_id}/revisions/{revision_id}",
        headers=get_headers(),
    )

    if response.status_code != 200:
        raise Exception(f"获取修订版本 ID {revision_id} 失败: {response.text}")

    return response.json()


def patch_deployment(deployment_id: str) -> None:
    """更新部署。"""
    headers = get_headers()
    headers["Content-Type"] = "application/json"

    # 这会创建一个新的修订版本,因为包含了 source_revision_config
    response = requests.patch(
        url=f"{CONTROL_PLANE_HOST}/v2/deployments/{deployment_id}",
        headers=headers,
        json={
            "source_config": {
                "build_on_push": True,
            },
            "source_revision_config": {
                "repo_ref": "main",
                "langgraph_config_path": "langgraph.json",
            },
        },
    )

    if response.status_code != 200:
        raise Exception(f"更新部署失败: {response.text}")

    print(f"已更新部署 ID {deployment_id}")


def wait_for_deployment(deployment_id: str, revision_id: str) -> None:
    """等待修订版本状态变为 DEPLOYED。"""
    start_time = time.time()
    revision, status = None, None
    while time.time() - start_time < MAX_WAIT_TIME:
        revision = get_revision(deployment_id, revision_id)
        status = revision["status"]
        if status == "DEPLOYED":
            break
        elif "FAILED" in status:
            raise Exception(f"修订版本 ID {revision_id} 失败: {revision}")

        print(f"等待修订版本 ID {revision_id} 变为 DEPLOYED...")
        time.sleep(60)

    if status != "DEPLOYED":
        raise Exception(
            f"等待修订版本 ID {revision_id} 变为 DEPLOYED 超时: {revision}"
        )


def delete_deployment(deployment_id: str) -> None:
    """删除部署。"""
    response = requests.delete(
        url=f"{CONTROL_PLANE_HOST}/v2/deployments/{deployment_id}",
        headers=get_headers(),
    )

    if response.status_code != 204:
        raise Exception(
            f"删除部署 ID {deployment_id} 失败: {response.text}"
        )

    print(f"部署 ID {deployment_id} 已删除")


if __name__ == "__main__":
    # 创建部署并获取最新修订版本
    deployment_id = create_deployment()
    revisions = list_revisions(deployment_id)
    latest_revision = revisions["resources"][0]
    latest_revision_id = latest_revision["id"]

    # 等待最新修订版本变为 DEPLOYED
    wait_for_deployment(deployment_id, latest_revision_id)

    # 更新部署并获取最新修订版本
    patch_deployment(deployment_id)
    revisions = list_revisions(deployment_id)
    latest_revision = revisions["resources"][0]
    latest_revision_id = latest_revision["id"]

    # 等待最新修订版本变为 DEPLOYED
    wait_for_deployment(deployment_id, latest_revision_id)

    # 删除部署
    delete_deployment(deployment_id)