dify/cli/src/version/probe.ts
L1nSn0w cfc1cf2b8c
refactor(cli/http): replace ky with a self-contained HTTP client (#36711)
Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-06-01 09:04:42 +00:00

137 lines
3.8 KiB
TypeScript

import type { ServerVersionResponse } from '@dify/contracts/api/openapi/types.gen'
import type { CompatVerdict } from './compat.js'
import type { Channel } from './info.js'
import type { HostsBundle } from '@/auth/hosts'
import { META_PROBE_TIMEOUT_MS, MetaClient } from '@/api/meta'
import { loadHosts } from '@/auth/hosts'
import { createHttpClient } from '@/http/client'
import { arch, platform } from '@/sys/index'
import { hostWithScheme, openAPIBase } from '@/util/host'
import { difyCompat, evaluateCompat } from './compat.js'
import { versionInfo } from './info.js'
export type ClientBlock = {
readonly version: string
readonly commit: string
readonly buildDate: string
readonly channel: Channel
readonly platform: string
readonly arch: string
}
export type ServerBlock = {
readonly endpoint: string
readonly reachable: boolean
readonly version?: string
readonly edition?: ServerVersionResponse['edition']
}
export type CompatBlock = CompatVerdict & {
readonly minDify: string
readonly maxDify: string
}
export type VersionReport = {
readonly client: ClientBlock
readonly server: ServerBlock
readonly compat: CompatBlock
}
// /openapi/v1/_version is intentionally unauthenticated, so the probe does not
// take a bearer. Same signature shape as the auto-nudge probe — easy to swap.
export type MetaProbe = (endpoint: string) => Promise<ServerVersionResponse>
export type RunVersionProbeOptions = {
readonly skipServer: boolean
readonly loadBundle?: () => Promise<HostsBundle | undefined>
readonly probe?: MetaProbe
}
const defaultLoadBundle = async (): Promise<HostsBundle | undefined> => loadHosts()
const defaultProbe: MetaProbe = async (endpoint) => {
const http = createHttpClient({ baseURL: openAPIBase(endpoint), timeoutMs: META_PROBE_TIMEOUT_MS, retryAttempts: 0 })
return new MetaClient(http).serverVersion()
}
function buildClientBlock(): ClientBlock {
return {
version: versionInfo.version,
commit: versionInfo.commit,
buildDate: versionInfo.buildDate,
channel: versionInfo.channel,
platform: platform(),
arch: arch(),
}
}
function unreachableServer(endpoint: string): ServerBlock {
return { endpoint, reachable: false }
}
function compatBlock(verdict: CompatVerdict): CompatBlock {
return {
minDify: difyCompat.minDify,
maxDify: difyCompat.maxDify,
status: verdict.status,
detail: verdict.detail,
}
}
export async function runVersionProbe(opts: RunVersionProbeOptions): Promise<VersionReport> {
const client = buildClientBlock()
if (opts.skipServer) {
return {
client,
server: { endpoint: '', reachable: false },
compat: compatBlock({ status: 'unknown', detail: 'server probe skipped' }),
}
}
const loadBundle = opts.loadBundle ?? defaultLoadBundle
const probe = opts.probe ?? defaultProbe
let bundle: HostsBundle | undefined
let loadFailed = false
try {
bundle = await loadBundle()
}
catch {
loadFailed = true
}
if (bundle === undefined || bundle.current_host === '') {
const detail = loadFailed ? 'hosts file unreadable' : 'no host configured'
return {
client,
server: { endpoint: '', reachable: false },
compat: compatBlock({ status: 'unknown', detail }),
}
}
const endpoint = hostWithScheme(bundle.current_host, bundle.scheme)
let serverInfo: ServerVersionResponse | undefined
try {
serverInfo = await probe(endpoint)
}
catch {
serverInfo = undefined
}
if (serverInfo === undefined)
return { client, server: unreachableServer(endpoint), compat: compatBlock({ status: 'unknown', detail: 'server unreachable' }) }
return {
client,
server: {
endpoint,
reachable: true,
version: serverInfo.version,
edition: serverInfo.edition,
},
compat: compatBlock(evaluateCompat(serverInfo.version)),
}
}