All checks were successful
build-prerelease / Resolve version stamps (push) Successful in 38s
CI / Format (push) Successful in 40s
CI / Clippy (push) Successful in 2m20s
build-prerelease / Build cortex binary (push) Successful in 4m25s
build-prerelease / Package cortex RPM (push) Successful in 1m22s
CI / Test (push) Successful in 4m34s
CI / Build cortex SRPM (push) Has been skipped
CI / Publish cortex to COPR (push) Has been skipped
CI / Build neuron SRPM (push) Has been skipped
CI / Publish neuron to COPR (push) Has been skipped
CI / Bump version in source (push) Has been skipped
build-prerelease / Build neuron-blackwell (push) Successful in 3m57s
build-prerelease / Build neuron-ampere (push) Successful in 4m51s
build-prerelease / Build neuron-ada (push) Successful in 5m12s
build-prerelease / Package helexa-neuron-ampere RPM (push) Successful in 2m49s
build-prerelease / Package helexa-neuron-ada RPM (push) Successful in 2m51s
build-prerelease / Package helexa-neuron-blackwell RPM (push) Successful in 3m43s
build-prerelease / Publish to rpm.lair.cafe (unstable) (push) Successful in 1m0s
Wires the in-flight TP machinery (Stage 7a workers, 7b-iii sharded
Qwen3) end to end so a non-streaming chat completion can run across
multiple GPUs via NCCL.
RPC additions (tp/rpc.rs):
- LoadDenseShard{model_id, config_json, safetensors_paths}
- GenerateStep{model_id, tokens, offset}
- ClearKvCache{model_id}
- UnloadModel{model_id}
- LoadDenseShardOk / GenerateStepOk / KvCacheCleared / Unloaded
Worker side (tp/worker.rs):
- WorkerState gains a `models: HashMap<String, TpQwen3ForCausalLM>`
keyed by model_id. LoadDenseShard mmaps safetensors via
ShardedVarBuilder (only this rank's slice materialises), builds the
TP model with the rank's NCCL Comm cloned from NcclState.
- GenerateStep runs the rank-local forward; the resulting logits are
dropped (only the leader's are used for sampling). The forward's
value here is the NCCL collectives inside the row-parallel layers
letting the leader's rank-0 forward make progress.
Pool side (tp/mod.rs):
- WorkerPool::load_dense_shard fans LoadDenseShard out to every worker,
builds rank 0's shard on the leader via spawn_blocking with a fresh
SendComm wrapper at the move boundary (Comm is !Send at the type
level), collects per-rank LoadDenseShardOk. Returns the leader's
Arc<Mutex<TpQwen3ForCausalLM>>.
- WorkerPool::generate_step fans GenerateStep out, runs the leader's
rank-0 forward in spawn_blocking (the AllReduce CustomOps inside
row-parallel layers block until every worker issues the matching
collective), returns the leader's last-position logits Tensor.
- WorkerPool::clear_kv_cache + unload_model follow the same pattern.
NcclState refactor (tp/nccl_state.rs):
- comm field becomes Option<Arc<Comm>> (was Option<Comm>) so callers
can share a clone with TpQwen3ForCausalLM::load.
- new `comm()` accessor + `SendComm` wrapper for spawn_blocking moves.
- single allow(clippy::arc_with_non_send_sync) at the canonical
construction site (Comm is !Send by type but the runtime invariant
is enforced by SendComm + the pool's Mutex).
Harness side (candle.rs):
- LoadedHandle enum (Single | Tp) replaces the bare Arc<LoadedModel>
in the harness's registry. list_models / unload_model /
inference_endpoint walk the enum uniformly.
- TpLoadedModel holds the pool + leader_model + tokenizer + devices.
- load_model dispatches on `spec.tensor_parallel > 1` to a new
cuda-gated load_tp path: resolve dense files via hf-hub, spawn the
pool, init_nccl, load_dense_shard.
- chat_completion branches on the handle variant. The TP path mirrors
run_inference: clear_kv_cache, prefill, sample, decode loop,
detokenize. Acquires the pool Mutex for the whole request.
- Streaming through TP is deferred to Stage 7c (returns Other(err)).
Script (script/validate-neuron.sh):
- 4th positional arg `tp_size` (default 1). When >1, switches to the
dense path (tp + GGUF is mutually exclusive — bails) and adds
`tensor_parallel` + `devices` to the load payload. NEURON_DEVICES
env overrides the default 0..N-1 device list.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
191 lines
7.2 KiB
Bash
Executable File
191 lines
7.2 KiB
Bash
Executable File
#!/bin/env bash
|
|
#
|
|
# End-to-end smoke test for a deployed neuron.
|
|
#
|
|
# Confirms the daemon is reachable, loads a small public Qwen3 GGUF,
|
|
# fires a reasoning probe at /v1/chat/completions, and prints the
|
|
# answer. Used to validate the candle harness on a real GPU host
|
|
# before trusting it for production traffic, and as a regression test
|
|
# after pushing new neuron builds.
|
|
#
|
|
# Usage:
|
|
# script/validate-neuron.sh [host] [model_id] [quant] [tp_size]
|
|
#
|
|
# Defaults:
|
|
# host = beast.hanzalova.internal
|
|
# model_id = unsloth/Qwen3-0.6B-GGUF (official Qwen3-*-GGUF repos
|
|
# ship Q8_0 only; unsloth's mirror ships the full Q-spectrum
|
|
# including Q4_K_M)
|
|
# quant = Q4_K_M (empty = dense safetensors path)
|
|
# tp_size = unset (= 1 = single-GPU; pass 2 to drive the TP path)
|
|
|
|
set -euo pipefail
|
|
|
|
HOST="${1:-beast.hanzalova.internal}"
|
|
MODEL_ID="${2:-unsloth/Qwen3-0.6B-GGUF}"
|
|
# `${3-Q4_K_M}` (no colon) only uses the default when the arg is
|
|
# UNSET — passing an explicit empty string drives the dense path.
|
|
QUANT="${3-Q4_K_M}"
|
|
# tp_size > 1 forces the dense path (TP requires safetensors) and adds
|
|
# `tensor_parallel: N` to the load payload. The harness picks device
|
|
# indices 0..N-1 by default; override by passing NEURON_DEVICES="0,1,..."
|
|
# in the environment.
|
|
TP_SIZE="${4-1}"
|
|
PORT="${NEURON_PORT:-13131}"
|
|
BASE="http://${HOST}:${PORT}"
|
|
|
|
# Reasoning probe — concrete, low-temperature answer that small models
|
|
# can still get right. "Paris" is a strong signal of basic competence
|
|
# beyond gibberish.
|
|
PROBE_PROMPT='What is the capital of Georgia (Caucasus)? Respond with the city name only, no punctuation.'
|
|
EXPECT_SUBSTR='Tbilisi'
|
|
# Qwen3 prepends <think>...</think> reasoning before the answer when the
|
|
# chat template enables thinking mode, which eats most of a small token
|
|
# budget. 256 leaves enough room for thinking + final answer.
|
|
MAX_TOKENS=256
|
|
|
|
# /models/load is synchronous — neuron blocks the response until the
|
|
# hf-hub download + GGUF parse + tensor materialisation is done. A
|
|
# fresh 0.6B-Q4_K_M is ~400 MB; on a slow link or cold cache that's
|
|
# easily a minute. Pick a generous ceiling.
|
|
LOAD_TIMEOUT=600
|
|
INFER_TIMEOUT=120
|
|
|
|
# Status messages go to stderr so command substitutions like
|
|
# `raw=$(run_probe)` capture only the function's intended return value
|
|
# (an HTTP body), not the progress chatter.
|
|
say() { printf '[%s] %s\n' "${HOST}" "$*" >&2; }
|
|
die() { say "FAIL: $*"; exit 1; }
|
|
|
|
probe_health() {
|
|
curl --silent --fail --max-time 5 "${BASE}/health" >/dev/null \
|
|
|| die "neuron not reachable at ${BASE}/health"
|
|
}
|
|
|
|
list_loaded_ids() {
|
|
# The manifest is YAML and uses yq; HTTP responses are JSON and use
|
|
# jq directly. pip-yq parses input as YAML by default, which trips
|
|
# on JSON content that happens to look like YAML aliases (chatcmpl
|
|
# ids, escaped quotes inside `<think>...</think>` blocks, etc.).
|
|
curl --silent --fail "${BASE}/models" | jq -r '.[].id'
|
|
}
|
|
|
|
is_loaded() {
|
|
list_loaded_ids 2>/dev/null | grep -Fxq "${MODEL_ID}"
|
|
}
|
|
|
|
trigger_load() {
|
|
# Build the per-rank CUDA device list as a JSON array. Either
|
|
# honour NEURON_DEVICES (`0,1,2`) verbatim or default to
|
|
# `[0, 1, ..., tp_size - 1]`.
|
|
local devices_json
|
|
if [[ -n "${NEURON_DEVICES:-}" ]]; then
|
|
devices_json=$(jq -n -c --arg s "${NEURON_DEVICES}" \
|
|
'$s | split(",") | map(tonumber)')
|
|
else
|
|
devices_json=$(jq -n -c --argjson n "${TP_SIZE}" '[range(0; $n)]')
|
|
fi
|
|
say "POST /models/load ${MODEL_ID} (quant=${QUANT:-<dense>}, tp=${TP_SIZE}, devices=${devices_json})"
|
|
say " (synchronous; may take a minute on first run while HF downloads)"
|
|
if (( TP_SIZE > 1 )) && [[ -n "${QUANT}" ]]; then
|
|
die "tp_size>1 requires dense safetensors — pass quant='' as the 3rd argument"
|
|
fi
|
|
# Build the payload via jq so the optional `quant` and
|
|
# `tensor_parallel` fields are omitted entirely when not in use —
|
|
# that's how the harness tells dense from quantized and single-GPU
|
|
# from TP.
|
|
local payload
|
|
if [[ -z "${QUANT}" ]] && (( TP_SIZE > 1 )); then
|
|
payload=$(jq -n -c \
|
|
--arg id "${MODEL_ID}" \
|
|
--argjson tp "${TP_SIZE}" \
|
|
--argjson devices "${devices_json}" \
|
|
'{model_id: $id, harness: "candle", tensor_parallel: $tp, devices: $devices}')
|
|
elif [[ -z "${QUANT}" ]]; then
|
|
payload=$(jq -n -c \
|
|
--arg id "${MODEL_ID}" \
|
|
--argjson devices "${devices_json}" \
|
|
'{model_id: $id, harness: "candle", devices: $devices}')
|
|
else
|
|
payload=$(jq -n -c \
|
|
--arg id "${MODEL_ID}" \
|
|
--arg q "${QUANT}" \
|
|
--argjson devices "${devices_json}" \
|
|
'{model_id: $id, harness: "candle", quant: $q, devices: $devices}')
|
|
fi
|
|
# --write-out captures the response code on a separate line so we
|
|
# can surface a real diagnostic instead of relying on --fail.
|
|
local resp http_code body
|
|
resp=$(curl --silent --show-error --max-time "${LOAD_TIMEOUT}" \
|
|
--write-out '\n__HTTP__%{http_code}' \
|
|
-X POST "${BASE}/models/load" \
|
|
-H 'content-type: application/json' \
|
|
--data "${payload}") || die "curl /models/load failed: $?"
|
|
http_code=$(echo "${resp}" | grep -oP '(?<=__HTTP__)\d+$' | tail -1)
|
|
body=$(echo "${resp}" | sed '$ s/__HTTP__.*$//')
|
|
if [[ "${http_code}" != "200" ]]; then
|
|
die "load returned HTTP ${http_code}: ${body}"
|
|
fi
|
|
say "load returned ${http_code}: ${body}"
|
|
}
|
|
|
|
run_probe() {
|
|
say "POST /v1/chat/completions (probe: ${PROBE_PROMPT})"
|
|
local payload
|
|
payload=$(jq -n -c \
|
|
--arg model "${MODEL_ID}" \
|
|
--arg content "${PROBE_PROMPT}" \
|
|
--argjson tokens "${MAX_TOKENS}" \
|
|
'{
|
|
model: $model,
|
|
messages: [{role: "user", content: $content}],
|
|
temperature: 0.1,
|
|
max_tokens: $tokens
|
|
}')
|
|
local resp http_code body
|
|
resp=$(curl --silent --show-error --max-time "${INFER_TIMEOUT}" \
|
|
--write-out '\n__HTTP__%{http_code}' \
|
|
-X POST "${BASE}/v1/chat/completions" \
|
|
-H 'content-type: application/json' \
|
|
--data "${payload}") || die "curl /v1/chat/completions failed: $?"
|
|
http_code=$(echo "${resp}" | grep -oP '(?<=__HTTP__)\d+$' | tail -1)
|
|
body=$(echo "${resp}" | sed '$ s/__HTTP__.*$//')
|
|
if [[ "${http_code}" != "200" ]]; then
|
|
die "inference returned HTTP ${http_code}: ${body}"
|
|
fi
|
|
echo "${body}"
|
|
}
|
|
|
|
say "validating neuron at ${BASE}"
|
|
probe_health
|
|
say "/health OK"
|
|
|
|
if is_loaded; then
|
|
say "${MODEL_ID} already loaded"
|
|
else
|
|
trigger_load
|
|
fi
|
|
|
|
raw=$(run_probe)
|
|
echo "---"
|
|
# Dump the raw JSON. Don't pipe through `yq -r '.'` — yq's default
|
|
# YAML output mode chokes on JSON strings that contain `<` (and the
|
|
# `<think>` markers Qwen3 emits during reasoning are a perfect
|
|
# example). The targeted `yq -r '.path'` calls below work fine
|
|
# because jq's path filter mode bypasses the YAML re-emit.
|
|
echo "${raw}"
|
|
echo "---"
|
|
|
|
content=$(echo "${raw}" | jq -r '.choices[0].message.content // empty')
|
|
if [[ -z "${content}" ]]; then
|
|
die "no content in chat completion response"
|
|
fi
|
|
say "assistant said: ${content}"
|
|
|
|
if echo "${content}" | grep -qiF "${EXPECT_SUBSTR}"; then
|
|
say "PASS — response contains expected substring '${EXPECT_SUBSTR}'"
|
|
exit 0
|
|
else
|
|
die "response did not contain '${EXPECT_SUBSTR}'"
|
|
fi
|