mirror of
https://github.com/bytedance/deer-flow.git
synced 2026-06-17 04:56:04 +00:00
Merge branch 'main' into 2.0.0-release
This commit is contained in:
@@ -412,7 +412,16 @@ def _provider_response(
|
|||||||
from app.gateway.auth_disabled import is_auth_disabled
|
from app.gateway.auth_disabled import is_auth_disabled
|
||||||
|
|
||||||
status, unavailable_reason = _provider_status(config, channels_config, provider)
|
status, unavailable_reason = _provider_status(config, channels_config, provider)
|
||||||
if connection:
|
if unavailable_reason is not None:
|
||||||
|
# The runtime provider is unavailable, so a stale "connected" row must
|
||||||
|
# not be reported as connected. Other statuses (e.g. "revoked") are
|
||||||
|
# preserved so consumers can still distinguish a revoked binding from a
|
||||||
|
# never-connected one.
|
||||||
|
if connection and connection["status"] != "connected":
|
||||||
|
connection_status = connection["status"]
|
||||||
|
else:
|
||||||
|
connection_status = "not_connected"
|
||||||
|
elif connection:
|
||||||
connection_status = connection["status"]
|
connection_status = connection["status"]
|
||||||
elif is_auth_disabled() and status["configured"] and unavailable_reason is None:
|
elif is_auth_disabled() and status["configured"] and unavailable_reason is None:
|
||||||
# Auth-disabled local mode routes every channel message to the default
|
# Auth-disabled local mode routes every channel message to the default
|
||||||
@@ -561,7 +570,6 @@ async def disconnect_channel_provider_runtime(provider: str, request: Request) -
|
|||||||
if not provider_config.enabled:
|
if not provider_config.enabled:
|
||||||
raise HTTPException(status_code=400, detail="Channel provider is not enabled")
|
raise HTTPException(status_code=400, detail="Channel provider is not enabled")
|
||||||
|
|
||||||
owner_user_id = _get_user_id(request)
|
|
||||||
try:
|
try:
|
||||||
repo = _get_repository(request, config)
|
repo = _get_repository(request, config)
|
||||||
except HTTPException as exc:
|
except HTTPException as exc:
|
||||||
@@ -569,25 +577,33 @@ async def disconnect_channel_provider_runtime(provider: str, request: Request) -
|
|||||||
raise
|
raise
|
||||||
repo = None
|
repo = None
|
||||||
|
|
||||||
if repo is not None:
|
current_channels_config = await _get_channels_config(request)
|
||||||
for connection in await repo.list_connections(owner_user_id):
|
candidate_channels_config = dict(current_channels_config)
|
||||||
if connection["provider"] == provider and connection["status"] != "revoked":
|
candidate_channels_config.pop(provider, None)
|
||||||
await repo.disconnect_connection(
|
|
||||||
connection_id=connection["id"],
|
|
||||||
owner_user_id=owner_user_id,
|
|
||||||
)
|
|
||||||
|
|
||||||
store = await _get_runtime_config_store(request)
|
stopped = await _sync_runtime_channel_after_removal(provider, candidate_channels_config)
|
||||||
await asyncio.to_thread(store.set_provider_disconnected, provider)
|
|
||||||
channels_config = await _load_channels_config(request, config)
|
|
||||||
request.app.state.channels_config = channels_config
|
|
||||||
|
|
||||||
stopped = await _sync_runtime_channel_after_removal(provider, channels_config)
|
|
||||||
if stopped is False:
|
if stopped is False:
|
||||||
display_name = _PROVIDER_META[provider]["display_name"]
|
display_name = _PROVIDER_META[provider]["display_name"]
|
||||||
raise HTTPException(status_code=400, detail=f"Failed to stop {display_name} channel. Try again.")
|
raise HTTPException(status_code=400, detail=f"Failed to stop {display_name} channel. Try again.")
|
||||||
|
|
||||||
return _provider_response(config, channels_config, provider, _PROVIDER_META[provider])
|
# Revoke the DB connection rows before committing the store/cache so a repo
|
||||||
|
# failure cannot leave the store and cache saying "disconnected" while the
|
||||||
|
# DB still holds "connected" rows that a later re-configure would silently
|
||||||
|
# reactivate.
|
||||||
|
if repo is not None:
|
||||||
|
await repo.disconnect_provider_connections(provider=provider)
|
||||||
|
|
||||||
|
store = await _get_runtime_config_store(request)
|
||||||
|
await asyncio.to_thread(store.set_provider_disconnected, provider)
|
||||||
|
|
||||||
|
# Re-read the live cached config and drop only this provider so a concurrent
|
||||||
|
# mutation for a different provider is not clobbered. No await may occur
|
||||||
|
# between this read and the reassignment.
|
||||||
|
live_channels_config = await _get_channels_config(request)
|
||||||
|
live_channels_config.pop(provider, None)
|
||||||
|
request.app.state.channels_config = live_channels_config
|
||||||
|
|
||||||
|
return _provider_response(config, live_channels_config, provider, _PROVIDER_META[provider])
|
||||||
|
|
||||||
|
|
||||||
@router.post("/{provider}/connect", response_model=ChannelConnectResponse)
|
@router.post("/{provider}/connect", response_model=ChannelConnectResponse)
|
||||||
@@ -656,8 +672,8 @@ async def configure_channel_provider_runtime(
|
|||||||
# cached by get_app_config().
|
# cached by get_app_config().
|
||||||
runtime_config["bot_username"] = values["bot_username"]
|
runtime_config["bot_username"] = values["bot_username"]
|
||||||
|
|
||||||
channels_config[provider] = runtime_config
|
candidate_channels_config = dict(channels_config)
|
||||||
request.app.state.channels_config = channels_config
|
candidate_channels_config[provider] = runtime_config
|
||||||
|
|
||||||
started = await _restart_runtime_channel_if_available(provider, runtime_config)
|
started = await _restart_runtime_channel_if_available(provider, runtime_config)
|
||||||
if started is False:
|
if started is False:
|
||||||
@@ -667,4 +683,11 @@ async def configure_channel_provider_runtime(
|
|||||||
store = await _get_runtime_config_store(request)
|
store = await _get_runtime_config_store(request)
|
||||||
await asyncio.to_thread(store.set_provider_config, provider, runtime_config)
|
await asyncio.to_thread(store.set_provider_config, provider, runtime_config)
|
||||||
|
|
||||||
return _provider_response(config, channels_config, provider, _PROVIDER_META[provider])
|
# Re-read the live cached config and apply only this provider's change so a
|
||||||
|
# concurrent mutation for a different provider is not clobbered. No await
|
||||||
|
# may occur between this read and the reassignment.
|
||||||
|
live_channels_config = await _get_channels_config(request)
|
||||||
|
live_channels_config[provider] = runtime_config
|
||||||
|
request.app.state.channels_config = live_channels_config
|
||||||
|
|
||||||
|
return _provider_response(config, live_channels_config, provider, _PROVIDER_META[provider])
|
||||||
|
|||||||
@@ -177,6 +177,24 @@ class ChannelConnectionRepository:
|
|||||||
await session.commit()
|
await session.commit()
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
async def disconnect_provider_connections(self, *, provider: str) -> int:
|
||||||
|
"""Revoke all active user connections for an instance-wide provider removal."""
|
||||||
|
async with self.session_factory() as session:
|
||||||
|
result = await session.execute(
|
||||||
|
select(ChannelConnectionRow.id).where(
|
||||||
|
ChannelConnectionRow.provider == provider,
|
||||||
|
ChannelConnectionRow.status != "revoked",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
connection_ids = [row_id for row_id in result.scalars()]
|
||||||
|
if not connection_ids:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
await session.execute(update(ChannelConnectionRow).where(ChannelConnectionRow.id.in_(connection_ids)).values(status="revoked"))
|
||||||
|
await session.execute(delete(ChannelCredentialRow).where(ChannelCredentialRow.connection_id.in_(connection_ids)))
|
||||||
|
await session.commit()
|
||||||
|
return len(connection_ids)
|
||||||
|
|
||||||
async def store_credentials(
|
async def store_credentials(
|
||||||
self,
|
self,
|
||||||
connection_id: str,
|
connection_id: str,
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ dependencies = [
|
|||||||
"deerflow-harness",
|
"deerflow-harness",
|
||||||
"fastapi>=0.115.0",
|
"fastapi>=0.115.0",
|
||||||
"httpx>=0.28.0",
|
"httpx>=0.28.0",
|
||||||
"python-multipart>=0.0.27",
|
"python-multipart>=0.0.31",
|
||||||
"sse-starlette>=2.1.0",
|
"sse-starlette>=2.1.0",
|
||||||
"uvicorn[standard]>=0.34.0",
|
"uvicorn[standard]>=0.34.0",
|
||||||
"lark-oapi>=1.4.0",
|
"lark-oapi>=1.4.0",
|
||||||
@@ -19,7 +19,7 @@ dependencies = [
|
|||||||
"wecom-aibot-python-sdk>=0.1.6",
|
"wecom-aibot-python-sdk>=0.1.6",
|
||||||
"dingtalk-stream>=0.24.3",
|
"dingtalk-stream>=0.24.3",
|
||||||
"bcrypt>=4.0.0",
|
"bcrypt>=4.0.0",
|
||||||
"pyjwt>=2.9.0",
|
"pyjwt>=2.13.0",
|
||||||
"email-validator>=2.0.0",
|
"email-validator>=2.0.0",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|||||||
@@ -294,6 +294,41 @@ def test_get_providers_reports_configured_channel_not_running(tmp_path, monkeypa
|
|||||||
anyio.run(repo.close)
|
anyio.run(repo.close)
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_providers_provider_unavailable_overrides_stale_connected_row(tmp_path):
|
||||||
|
import anyio
|
||||||
|
|
||||||
|
repo = anyio.run(_make_repo, tmp_path)
|
||||||
|
|
||||||
|
async def seed_connection():
|
||||||
|
await repo.upsert_connection(
|
||||||
|
owner_user_id=str(_user().id),
|
||||||
|
provider="slack",
|
||||||
|
external_account_id="U123",
|
||||||
|
status="connected",
|
||||||
|
)
|
||||||
|
|
||||||
|
anyio.run(seed_connection)
|
||||||
|
config = ChannelConnectionsConfig.model_validate(
|
||||||
|
{
|
||||||
|
"enabled": True,
|
||||||
|
"slack": {"enabled": True},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
app = _make_app(config, repo, {})
|
||||||
|
|
||||||
|
with TestClient(app) as client:
|
||||||
|
response = client.get("/api/channels/providers")
|
||||||
|
|
||||||
|
assert response.status_code == 200
|
||||||
|
by_provider = {item["provider"]: item for item in response.json()["providers"]}
|
||||||
|
assert by_provider["slack"]["configured"] is False
|
||||||
|
assert by_provider["slack"]["connectable"] is False
|
||||||
|
assert by_provider["slack"]["connection_status"] == "not_connected"
|
||||||
|
assert "Slack credentials" in by_provider["slack"]["unavailable_reason"]
|
||||||
|
|
||||||
|
anyio.run(repo.close)
|
||||||
|
|
||||||
|
|
||||||
def test_get_providers_restarts_configured_channel_when_service_can_reconcile(tmp_path, monkeypatch):
|
def test_get_providers_restarts_configured_channel_when_service_can_reconcile(tmp_path, monkeypatch):
|
||||||
import anyio
|
import anyio
|
||||||
|
|
||||||
@@ -614,6 +649,54 @@ def test_runtime_config_endpoints_require_admin(tmp_path):
|
|||||||
anyio.run(repo.close)
|
anyio.run(repo.close)
|
||||||
|
|
||||||
|
|
||||||
|
def test_configure_provider_runtime_rolls_back_visible_state_when_start_fails(tmp_path, monkeypatch):
|
||||||
|
import anyio
|
||||||
|
|
||||||
|
repo = anyio.run(_make_repo, tmp_path)
|
||||||
|
config = ChannelConnectionsConfig.model_validate(
|
||||||
|
{
|
||||||
|
"enabled": True,
|
||||||
|
"slack": {"enabled": True},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
existing_runtime_config = {
|
||||||
|
"enabled": True,
|
||||||
|
"bot_token": "xoxb-old",
|
||||||
|
"app_token": "xapp-old",
|
||||||
|
}
|
||||||
|
runtime_config_store = ChannelRuntimeConfigStore(tmp_path / "channels" / "runtime-config.json")
|
||||||
|
runtime_config_store.set_provider_config("slack", existing_runtime_config)
|
||||||
|
service = SimpleNamespace(configure_channel=AsyncMock(return_value=False))
|
||||||
|
monkeypatch.setattr("app.channels.service.get_channel_service", lambda: service)
|
||||||
|
app = _make_app(
|
||||||
|
config,
|
||||||
|
repo,
|
||||||
|
{"slack": dict(existing_runtime_config)},
|
||||||
|
runtime_config_store=runtime_config_store,
|
||||||
|
)
|
||||||
|
|
||||||
|
with TestClient(app) as client:
|
||||||
|
response = client.post(
|
||||||
|
"/api/channels/slack/runtime-config",
|
||||||
|
json={"values": {"bot_token": "xoxb-new", "app_token": "xapp-new"}},
|
||||||
|
)
|
||||||
|
|
||||||
|
assert response.status_code == 400
|
||||||
|
assert "Failed to start Slack channel" in response.json()["detail"]
|
||||||
|
service.configure_channel.assert_awaited_once_with(
|
||||||
|
"slack",
|
||||||
|
{
|
||||||
|
"enabled": True,
|
||||||
|
"bot_token": "xoxb-new",
|
||||||
|
"app_token": "xapp-new",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
assert app.state.channels_config["slack"] == existing_runtime_config
|
||||||
|
assert runtime_config_store.get_provider_config("slack") == existing_runtime_config
|
||||||
|
|
||||||
|
anyio.run(repo.close)
|
||||||
|
|
||||||
|
|
||||||
def test_configure_telegram_runtime_uses_new_bot_username_for_deep_link_without_mutating_config(tmp_path):
|
def test_configure_telegram_runtime_uses_new_bot_username_for_deep_link_without_mutating_config(tmp_path):
|
||||||
import anyio
|
import anyio
|
||||||
|
|
||||||
@@ -862,7 +945,7 @@ def test_disconnect_provider_runtime_config_suppresses_file_config_and_stops_cha
|
|||||||
anyio.run(repo.close)
|
anyio.run(repo.close)
|
||||||
|
|
||||||
|
|
||||||
def test_disconnect_provider_runtime_config_revokes_current_user_provider_connections(tmp_path):
|
def test_disconnect_provider_runtime_config_revokes_all_provider_connections(tmp_path):
|
||||||
import anyio
|
import anyio
|
||||||
|
|
||||||
repo = anyio.run(_make_repo, tmp_path)
|
repo = anyio.run(_make_repo, tmp_path)
|
||||||
@@ -874,6 +957,18 @@ def test_disconnect_provider_runtime_config_revokes_current_user_provider_connec
|
|||||||
external_account_id="U123",
|
external_account_id="U123",
|
||||||
status="connected",
|
status="connected",
|
||||||
)
|
)
|
||||||
|
await repo.upsert_connection(
|
||||||
|
owner_user_id="other-user",
|
||||||
|
provider="slack",
|
||||||
|
external_account_id="U456",
|
||||||
|
status="connected",
|
||||||
|
)
|
||||||
|
await repo.upsert_connection(
|
||||||
|
owner_user_id="other-user",
|
||||||
|
provider="telegram",
|
||||||
|
external_account_id="42",
|
||||||
|
status="connected",
|
||||||
|
)
|
||||||
|
|
||||||
anyio.run(seed_connection)
|
anyio.run(seed_connection)
|
||||||
config = ChannelConnectionsConfig.model_validate(
|
config = ChannelConnectionsConfig.model_validate(
|
||||||
@@ -895,10 +990,132 @@ def test_disconnect_provider_runtime_config_revokes_current_user_provider_connec
|
|||||||
assert configure_response.status_code == 200
|
assert configure_response.status_code == 200
|
||||||
assert disconnect_response.status_code == 200
|
assert disconnect_response.status_code == 200
|
||||||
|
|
||||||
async def get_connection_status():
|
async def get_connection_statuses():
|
||||||
return (await repo.list_connections(str(_user().id)))[0]["status"]
|
return {
|
||||||
|
"admin_slack": (await repo.list_connections(str(_user().id)))[0]["status"],
|
||||||
|
"other": {item["provider"]: item["status"] for item in await repo.list_connections("other-user")},
|
||||||
|
}
|
||||||
|
|
||||||
assert anyio.run(get_connection_status) == "revoked"
|
statuses = anyio.run(get_connection_statuses)
|
||||||
|
assert statuses["admin_slack"] == "revoked"
|
||||||
|
assert statuses["other"]["slack"] == "revoked"
|
||||||
|
assert statuses["other"]["telegram"] == "connected"
|
||||||
|
|
||||||
|
anyio.run(repo.close)
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_providers_preserves_revoked_status_when_provider_unavailable(tmp_path):
|
||||||
|
import anyio
|
||||||
|
|
||||||
|
repo = anyio.run(_make_repo, tmp_path)
|
||||||
|
|
||||||
|
async def seed_connection():
|
||||||
|
await repo.upsert_connection(
|
||||||
|
owner_user_id=str(_user().id),
|
||||||
|
provider="slack",
|
||||||
|
external_account_id="U123",
|
||||||
|
status="revoked",
|
||||||
|
)
|
||||||
|
|
||||||
|
anyio.run(seed_connection)
|
||||||
|
config = ChannelConnectionsConfig.model_validate(
|
||||||
|
{
|
||||||
|
"enabled": True,
|
||||||
|
"slack": {"enabled": True},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
# No runtime channels_config -> the slack provider is unavailable.
|
||||||
|
app = _make_app(config, repo, {})
|
||||||
|
|
||||||
|
with TestClient(app) as client:
|
||||||
|
response = client.get("/api/channels/providers")
|
||||||
|
|
||||||
|
assert response.status_code == 200
|
||||||
|
by_provider = {item["provider"]: item for item in response.json()["providers"]}
|
||||||
|
assert by_provider["slack"]["connectable"] is False
|
||||||
|
assert by_provider["slack"]["unavailable_reason"] is not None
|
||||||
|
# A revoked binding must stay distinguishable from a never-connected one,
|
||||||
|
# even when the runtime provider is currently unavailable.
|
||||||
|
assert by_provider["slack"]["connection_status"] == "revoked"
|
||||||
|
|
||||||
|
anyio.run(repo.close)
|
||||||
|
|
||||||
|
|
||||||
|
def test_configure_provider_runtime_does_not_clobber_concurrent_config_update(tmp_path, monkeypatch):
|
||||||
|
import anyio
|
||||||
|
|
||||||
|
repo = anyio.run(_make_repo, tmp_path)
|
||||||
|
config = ChannelConnectionsConfig.model_validate(
|
||||||
|
{
|
||||||
|
"enabled": True,
|
||||||
|
"slack": {"enabled": True},
|
||||||
|
"telegram": {"enabled": True, "bot_username": "deerflow_bot"},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
runtime_config_store = ChannelRuntimeConfigStore(tmp_path / "channels" / "runtime-config.json")
|
||||||
|
app = _make_app(config, repo, {}, runtime_config_store=runtime_config_store)
|
||||||
|
|
||||||
|
async def configure_channel(provider, runtime_config):
|
||||||
|
# Simulate a concurrent admin request for a *different* provider whose
|
||||||
|
# write to app.state lands while this request awaits the worker restart.
|
||||||
|
app.state.channels_config = {
|
||||||
|
**app.state.channels_config,
|
||||||
|
"telegram": {"enabled": True, "bot_token": "tg-token"},
|
||||||
|
}
|
||||||
|
return True
|
||||||
|
|
||||||
|
service = SimpleNamespace(configure_channel=configure_channel)
|
||||||
|
monkeypatch.setattr("app.channels.service.get_channel_service", lambda: service)
|
||||||
|
|
||||||
|
with TestClient(app) as client:
|
||||||
|
response = client.post(
|
||||||
|
"/api/channels/slack/runtime-config",
|
||||||
|
json={"values": {"bot_token": "xoxb-ui", "app_token": "xapp-ui"}},
|
||||||
|
)
|
||||||
|
|
||||||
|
assert response.status_code == 200
|
||||||
|
# The concurrent telegram write must survive alongside the slack write.
|
||||||
|
assert app.state.channels_config["slack"]["bot_token"] == "xoxb-ui"
|
||||||
|
assert app.state.channels_config["telegram"]["bot_token"] == "tg-token"
|
||||||
|
|
||||||
|
anyio.run(repo.close)
|
||||||
|
|
||||||
|
|
||||||
|
def test_disconnect_provider_runtime_keeps_state_consistent_when_revoke_fails(tmp_path):
|
||||||
|
import anyio
|
||||||
|
|
||||||
|
repo = anyio.run(_make_repo, tmp_path)
|
||||||
|
config = ChannelConnectionsConfig.model_validate(
|
||||||
|
{
|
||||||
|
"enabled": True,
|
||||||
|
"slack": {"enabled": True},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
runtime_config_store = ChannelRuntimeConfigStore(tmp_path / "channels" / "runtime-config.json")
|
||||||
|
app = _make_app(config, repo, {}, runtime_config_store=runtime_config_store)
|
||||||
|
|
||||||
|
with TestClient(app) as client:
|
||||||
|
configure_response = client.post(
|
||||||
|
"/api/channels/slack/runtime-config",
|
||||||
|
json={"values": {"bot_token": "xoxb-ui", "app_token": "xapp-ui"}},
|
||||||
|
)
|
||||||
|
assert configure_response.status_code == 200
|
||||||
|
|
||||||
|
repo.disconnect_provider_connections = AsyncMock(side_effect=RuntimeError("db down"))
|
||||||
|
|
||||||
|
with TestClient(app, raise_server_exceptions=False) as client:
|
||||||
|
disconnect_response = client.delete("/api/channels/slack/runtime-config")
|
||||||
|
|
||||||
|
assert disconnect_response.status_code == 500
|
||||||
|
# When the DB revoke fails, the store/cache must not be left diverged from
|
||||||
|
# the DB: the provider stays configured so a later re-configure cannot
|
||||||
|
# silently reactivate un-revoked connection rows.
|
||||||
|
assert app.state.channels_config["slack"]["bot_token"] == "xoxb-ui"
|
||||||
|
assert runtime_config_store.get_provider_config("slack") == {
|
||||||
|
"enabled": True,
|
||||||
|
"bot_token": "xoxb-ui",
|
||||||
|
"app_token": "xapp-ui",
|
||||||
|
}
|
||||||
|
|
||||||
anyio.run(repo.close)
|
anyio.run(repo.close)
|
||||||
|
|
||||||
|
|||||||
Generated
+8
-8
@@ -807,8 +807,8 @@ requires-dist = [
|
|||||||
{ name = "langgraph-sdk", specifier = ">=0.1.51" },
|
{ name = "langgraph-sdk", specifier = ">=0.1.51" },
|
||||||
{ name = "lark-oapi", specifier = ">=1.4.0" },
|
{ name = "lark-oapi", specifier = ">=1.4.0" },
|
||||||
{ name = "markdown-to-mrkdwn", specifier = ">=0.3.1" },
|
{ name = "markdown-to-mrkdwn", specifier = ">=0.3.1" },
|
||||||
{ name = "pyjwt", specifier = ">=2.9.0" },
|
{ name = "pyjwt", specifier = ">=2.13.0" },
|
||||||
{ name = "python-multipart", specifier = ">=0.0.27" },
|
{ name = "python-multipart", specifier = ">=0.0.31" },
|
||||||
{ name = "python-telegram-bot", specifier = ">=21.0" },
|
{ name = "python-telegram-bot", specifier = ">=21.0" },
|
||||||
{ name = "slack-sdk", specifier = ">=3.33.0" },
|
{ name = "slack-sdk", specifier = ">=3.33.0" },
|
||||||
{ name = "sse-starlette", specifier = ">=2.1.0" },
|
{ name = "sse-starlette", specifier = ">=2.1.0" },
|
||||||
@@ -3417,11 +3417,11 @@ wheels = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pyjwt"
|
name = "pyjwt"
|
||||||
version = "2.12.1"
|
version = "2.13.0"
|
||||||
source = { registry = "https://pypi.org/simple" }
|
source = { registry = "https://pypi.org/simple" }
|
||||||
sdist = { url = "https://files.pythonhosted.org/packages/c2/27/a3b6e5bf6ff856d2509292e95c8f57f0df7017cf5394921fc4e4ef40308a/pyjwt-2.12.1.tar.gz", hash = "sha256:c74a7a2adf861c04d002db713dd85f84beb242228e671280bf709d765b03672b", size = 102564, upload-time = "2026-03-13T19:27:37.25Z" }
|
sdist = { url = "https://files.pythonhosted.org/packages/3b/81/58d0ac84e1ef3a3843791d6954d94c0b33d526c75eeb1efbce9d0a4c4077/pyjwt-2.13.0.tar.gz", hash = "sha256:41571c89ca91598c79e8ef18a2d07367d4810fbbd6f637794879baf1b7703423", size = 107515, upload-time = "2026-05-21T19:54:36.618Z" }
|
||||||
wheels = [
|
wheels = [
|
||||||
{ url = "https://files.pythonhosted.org/packages/e5/7a/8dd906bd22e79e47397a61742927f6747fe93242ef86645ee9092e610244/pyjwt-2.12.1-py3-none-any.whl", hash = "sha256:28ca37c070cad8ba8cd9790cd940535d40274d22f80ab87f3ac6a713e6e8454c", size = 29726, upload-time = "2026-03-13T19:27:35.677Z" },
|
{ url = "https://files.pythonhosted.org/packages/a3/5e/ecf12fdb62546d64385c158514e9b2b671f7832108ef2ecd2020ce0af2d1/pyjwt-2.13.0-py3-none-any.whl", hash = "sha256:66adcc2aff09b3f1bbd95fc1e1577df8ac8723c978552fd43304c8a290ac5728", size = 31274, upload-time = "2026-05-21T19:54:35.362Z" },
|
||||||
]
|
]
|
||||||
|
|
||||||
[package.optional-dependencies]
|
[package.optional-dependencies]
|
||||||
@@ -3568,11 +3568,11 @@ wheels = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "python-multipart"
|
name = "python-multipart"
|
||||||
version = "0.0.27"
|
version = "0.0.31"
|
||||||
source = { registry = "https://pypi.org/simple" }
|
source = { registry = "https://pypi.org/simple" }
|
||||||
sdist = { url = "https://files.pythonhosted.org/packages/69/9b/f23807317a113dc36e74e75eb265a02dd1a4d9082abc3c1064acd22997c4/python_multipart-0.0.27.tar.gz", hash = "sha256:9870a6a8c5a20a5bf4f07c017bd1489006ff8836cff097b6933355ee2b49b602", size = 44043, upload-time = "2026-04-27T10:51:26.649Z" }
|
sdist = { url = "https://files.pythonhosted.org/packages/64/7e/9b35ad8f3d9ca680f7c87a88f19612fdd8da9796c4d3b46e560ac79dcc4a/python_multipart-0.0.31.tar.gz", hash = "sha256:fc631183bb13e56db3158a4909908dfb2e23565286744e798241e63750e5d680", size = 46689, upload-time = "2026-06-04T08:27:49.014Z" }
|
||||||
wheels = [
|
wheels = [
|
||||||
{ url = "https://files.pythonhosted.org/packages/99/78/4126abcbdbd3c559d43e0db7f7b9173fc6befe45d39a2856cc0b8ec2a5a6/python_multipart-0.0.27-py3-none-any.whl", hash = "sha256:6fccfad17a27334bd0193681b369f476eda3409f17381a2d65aa7df3f7275645", size = 29254, upload-time = "2026-04-27T10:51:24.997Z" },
|
{ url = "https://files.pythonhosted.org/packages/5e/1e/7f7f299527a5a8ad90acd5f2f78dfa6c8495c6301a3205106ea68a84de96/python_multipart-0.0.31-py3-none-any.whl", hash = "sha256:8408153d68a9773291fc1da39a8b85a50044bddbabd2dd72e9229776b7b15e28", size = 29996, upload-time = "2026-06-04T08:27:47.804Z" },
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
|||||||
@@ -117,7 +117,9 @@ export function WorkspaceChannelsList() {
|
|||||||
<SidebarMenu>
|
<SidebarMenu>
|
||||||
{visibleProviders.map((provider) => {
|
{visibleProviders.map((provider) => {
|
||||||
const canEditRuntimeConfig = providerCanEditRuntimeConfig(provider);
|
const canEditRuntimeConfig = providerCanEditRuntimeConfig(provider);
|
||||||
const isConnected = provider.connection_status === "connected";
|
const isConnected =
|
||||||
|
!provider.unavailable_reason &&
|
||||||
|
provider.connection_status === "connected";
|
||||||
const isPending =
|
const isPending =
|
||||||
(connectMutation.isPending &&
|
(connectMutation.isPending &&
|
||||||
connectMutation.variables === provider.provider) ||
|
connectMutation.variables === provider.provider) ||
|
||||||
|
|||||||
@@ -117,9 +117,11 @@ function ChannelProviderItem({
|
|||||||
const configureMutation = useConfigureChannelProvider();
|
const configureMutation = useConfigureChannelProvider();
|
||||||
const disconnectProviderMutation = useDisconnectChannelProvider();
|
const disconnectProviderMutation = useDisconnectChannelProvider();
|
||||||
const [setupOpen, setSetupOpen] = useState(false);
|
const [setupOpen, setSetupOpen] = useState(false);
|
||||||
|
const runtimeAvailable = provider.configured && !provider.unavailable_reason;
|
||||||
const isConnected =
|
const isConnected =
|
||||||
connection?.status === "connected" ||
|
runtimeAvailable &&
|
||||||
provider.connection_status === "connected";
|
(connection?.status === "connected" ||
|
||||||
|
provider.connection_status === "connected");
|
||||||
const canEditRuntimeConfig = providerCanEditRuntimeConfig(provider);
|
const canEditRuntimeConfig = providerCanEditRuntimeConfig(provider);
|
||||||
const canConnect =
|
const canConnect =
|
||||||
(provider.connectable ?? (provider.enabled && provider.configured)) &&
|
(provider.connectable ?? (provider.enabled && provider.configured)) &&
|
||||||
@@ -186,7 +188,7 @@ function ChannelProviderItem({
|
|||||||
</ItemTitle>
|
</ItemTitle>
|
||||||
<ItemDescription className="line-clamp-none">
|
<ItemDescription className="line-clamp-none">
|
||||||
{getProviderDescription(provider, t.channels.descriptions)}
|
{getProviderDescription(provider, t.channels.descriptions)}
|
||||||
{connectionLabel
|
{isConnected && connectionLabel
|
||||||
? ` ${t.channels.connectedAs(connectionLabel)}`
|
? ` ${t.channels.connectedAs(connectionLabel)}`
|
||||||
: ""}
|
: ""}
|
||||||
{!isConnected && provider.unavailable_reason
|
{!isConnected && provider.unavailable_reason
|
||||||
|
|||||||
Reference in New Issue
Block a user