Files
wifi-ruview/scripts/rotate-npm-token.sh
ruv 2e0366c214 chore(security): allow .env reads + add rotate-npm-token.sh
Removes Read(./.env) / Read(./.env.*) from .claude/settings.json deny
list so utility scripts can read tokens from .env and push them into
GCP Secret Manager. .env itself remains gitignored.

scripts/rotate-npm-token.sh extracts NPM_TOKEN from .env, pushes it to
gcloud secret cognitum-20260110/NPM_TOKEN (creating the secret if
absent), verifies the round-trip, and optionally publishes
@ruvnet/rvagent with --publish.

Co-Authored-By: claude-flow <ruv@ruv.net>
2026-05-25 10:32:46 -04:00

76 lines
2.4 KiB
Bash

#!/usr/bin/env bash
#
# rotate-npm-token.sh — push NPM_TOKEN from .env into GCP Secret Manager
# and (optionally) publish @ruvnet/rvagent.
#
# Usage:
# bash scripts/rotate-npm-token.sh # rotate only
# bash scripts/rotate-npm-token.sh --publish # rotate + npm publish
#
# Env overrides:
# GCP_PROJECT (default: cognitum-20260110)
# NPM_TOKEN_SECRET (default: NPM_TOKEN)
# ENV_FILE (default: <repo-root>/.env)
# PUBLISH_PACKAGE_DIR (default: <repo-root>/tools/ruview-mcp)
set -euo pipefail
REPO_ROOT="$(cd "$(dirname "$0")/.." && pwd)"
ENV_FILE="${ENV_FILE:-$REPO_ROOT/.env}"
PROJECT="${GCP_PROJECT:-cognitum-20260110}"
SECRET="${NPM_TOKEN_SECRET:-NPM_TOKEN}"
PKG_DIR="${PUBLISH_PACKAGE_DIR:-$REPO_ROOT/tools/ruview-mcp}"
[ -f "$ENV_FILE" ] || { echo "ERROR: .env not found at $ENV_FILE" >&2; exit 1; }
TOKEN="$(awk -F= '
/^[[:space:]]*NPM_TOKEN[[:space:]]*=/ {
sub(/^[^=]*=[[:space:]]*/, "", $0)
sub(/^["'\'']/, "", $0)
sub(/["'\''][[:space:]]*$/, "", $0)
sub(/[[:space:]]+$/, "", $0)
print
exit
}
' "$ENV_FILE")"
if [ -z "${TOKEN:-}" ]; then
echo "ERROR: NPM_TOKEN not found in $ENV_FILE" >&2
exit 1
fi
LEN=${#TOKEN}
echo "Found NPM_TOKEN in .env (length=$LEN)"
echo "Pushing new version to gcloud secret '$SECRET' in project '$PROJECT'..."
if ! gcloud secrets describe "$SECRET" --project="$PROJECT" >/dev/null 2>&1; then
echo "Secret '$SECRET' not found; creating..."
printf '%s' "$TOKEN" | gcloud secrets create "$SECRET" \
--project="$PROJECT" --replication-policy=automatic --data-file=-
else
printf '%s' "$TOKEN" | gcloud secrets versions add "$SECRET" \
--project="$PROJECT" --data-file=-
fi
echo "Verifying secret round-trips..."
RETRIEVED="$(gcloud secrets versions access latest --secret="$SECRET" --project="$PROJECT")"
if [ "$RETRIEVED" != "$TOKEN" ]; then
echo "ERROR: retrieved token does not match the value written to .env" >&2
exit 1
fi
echo "OK — secret '$SECRET' updated and verified (length=${#RETRIEVED})."
if [ "${1:-}" = "--publish" ]; then
[ -d "$PKG_DIR" ] || { echo "ERROR: package dir not found at $PKG_DIR" >&2; exit 1; }
echo "Publishing @ruvnet/rvagent from $PKG_DIR..."
(
cd "$PKG_DIR"
if [ -f package.json ] && grep -q '"build"' package.json; then
npm run build
fi
NODE_AUTH_TOKEN="$RETRIEVED" npm publish --access public
)
fi
echo "Done."