#!/usr/bin/env bash # --- # name: ppt-setup # author: Z.AI # version: "1.0" # description: Environment setup for the PPT skill. Checks and installs all required dependencies. # --- # # Installs only dependencies required by the PPT skill. set -euo pipefail RED='\033[0;31m'; GREEN='\033[0;32m'; YELLOW='\033[1;33m'; BLUE='\033[0;34m'; NC='\033[0m' ok() { echo -e " ${GREEN}✓${NC} $1"; } fail() { echo -e " ${RED}✗${NC} $1"; } warn() { echo -e " ${YELLOW}○${NC} $1"; } info() { echo -e " ${BLUE}→${NC} $1"; } echo "============================================" echo " PPT Skill — Environment Setup" echo "============================================" echo "" OS="$(uname -s)" ARCH="$(uname -m)" echo "Platform: $OS $ARCH" echo "" # ── 0. macOS: Homebrew ── if [ "$OS" = "Darwin" ]; then echo "--- Homebrew (macOS package manager) ---" if command -v brew &>/dev/null; then BREW_VER=$(brew --version 2>/dev/null | head -1) ok "brew ($BREW_VER)" else fail "brew not found — Node.js install needs Homebrew on macOS" info "Install: /bin/bash -c \"\$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)\"" fi echo "" fi # ── 1. Node.js (PptxGenJS runs on Node) ── echo "--- Node.js ---" if command -v node &>/dev/null; then NODE_VER=$(node --version) ok "node ($NODE_VER)" else fail "node not found (required — PPTX generation uses PptxGenJS on Node)" case "$OS" in Darwin) info "Install: brew install node" ;; Linux) info "Install: curl -fsSL https://deb.nodesource.com/setup_20.x | sudo -E bash -" info " sudo apt install -y nodejs" ;; *) info "Install: https://nodejs.org/" ;; esac fi # ── 2. npm ── echo "" echo "--- npm ---" if command -v npm &>/dev/null; then NPM_VER=$(npm --version 2>/dev/null) ok "npm ($NPM_VER)" else fail "npm not found" case "$OS" in Darwin) info "Install: brew install node (includes npm)" ;; Linux) info "Install: comes with nodejs" ;; *) info "Install: https://nodejs.org/" ;; esac fi # ── 3. npm package: pptxgenjs ── echo "" echo "--- npm Packages ---" if node -e "require('pptxgenjs')" 2>/dev/null || npm list -g pptxgenjs &>/dev/null; then PPTX_VER=$(node -e "try{console.log(require('pptxgenjs/package.json').version)}catch(e){console.log('installed')}" 2>/dev/null) ok "pptxgenjs ($PPTX_VER)" else fail "pptxgenjs not installed" info "Install: npm install -g pptxgenjs" echo "" if [ -t 0 ]; then read -p " Install now? [Y/n] " -n 1 -r REPLY echo "" REPLY=${REPLY:-Y} else warn "Non-interactive mode — skipping auto-install." REPLY=N fi if [[ ! $REPLY =~ ^[Nn]$ ]]; then npm install -g pptxgenjs 2>/dev/null && ok "Installed: pptxgenjs" || fail "npm install failed" fi fi # ── 4. Python 3 (OOXML validation scripts) ── echo "" echo "--- Python (OOXML validation) ---" if command -v python3 &>/dev/null; then PY_VER=$(python3 --version 2>&1) ok "python3 ($PY_VER)" if [ "$OS" = "Darwin" ]; then PY_PATH=$(which python3 2>/dev/null) if [[ "$PY_PATH" == "/usr/bin/python3" ]]; then warn "Using macOS system Python (limited). Recommend: brew install python3" fi fi else fail "python3 not found" case "$OS" in Darwin) info "Install: brew install python3" ;; Linux) info "Install: sudo apt install python3 python3-pip (Debian/Ubuntu)" info " sudo dnf install python3 python3-pip (Fedora/RHEL)" ;; *) info "Install: https://www.python.org/downloads/" ;; esac fi # ── 5. pip ── echo "" echo "--- pip ---" if python3 -m pip --version &>/dev/null 2>&1; then PIP_VER=$(python3 -m pip --version 2>/dev/null | head -1) ok "pip ($PIP_VER)" else fail "pip not found" case "$OS" in Darwin) info "Install: python3 -m ensurepip --upgrade" info " or: brew install python3 (includes pip)" ;; Linux) info "Install: sudo apt install python3-pip (Debian/Ubuntu)" ;; *) info "Install: python3 -m ensurepip --upgrade" ;; esac fi # ── 6. Python packages ── echo "" echo "--- Python Packages ---" PY_PKGS=( "lxml:lxml" "defusedxml:defusedxml" ) MISSING_PY=() for entry in "${PY_PKGS[@]}"; do mod="${entry%%:*}" pkg="${entry##*:}" if python3 -c "import $mod" 2>/dev/null; then ver=$(python3 -c "import $mod; print(getattr($mod, '__version__', 'installed'))" 2>/dev/null) ok "$pkg ($ver)" else fail "$pkg not installed" MISSING_PY+=("$pkg") fi done if [ ${#MISSING_PY[@]} -gt 0 ]; then echo "" if [ -t 0 ]; then read -p " Install missing Python packages? [Y/n] " -n 1 -r REPLY echo "" REPLY=${REPLY:-Y} else warn "Non-interactive mode — skipping auto-install. Run interactively or install manually." REPLY=N fi if [[ ! $REPLY =~ ^[Nn]$ ]]; then python3 -m pip install -q "${MISSING_PY[@]}" 2>/dev/null \ || python3 -m pip install -q --user "${MISSING_PY[@]}" 2>/dev/null \ || python3 -m pip install -q --break-system-packages "${MISSING_PY[@]}" 2>/dev/null \ || { fail "pip install failed. Try manually: pip install ${MISSING_PY[*]}"; } ok "Installed: ${MISSING_PY[*]}" fi fi # ── 7. Tectonic (optional, for Beamer/LaTeX slides) ── echo "" echo "--- Tectonic (optional, Beamer slides only) ---" if command -v tectonic &>/dev/null; then TEC_VER=$(tectonic --version 2>&1 | head -1) ok "tectonic ($TEC_VER)" elif [ -x "$HOME/tectonic" ]; then ok "tectonic (~/tectonic)" else warn "tectonic not installed (needed only for Beamer/LaTeX slide PDFs)" case "$OS" in Darwin) info "Install: brew install tectonic" ;; Linux) info "Install: conda install -c conda-forge tectonic" info " or: curl -fsSL https://drop-sh.fullyjustified.net | sh" ;; MINGW*|MSYS*|CYGWIN*) info "Install: scoop install tectonic" ;; esac fi # ── Summary ── echo "" echo "============================================" echo " Setup complete." echo " Core: Node.js + pptxgenjs" echo " Validation: Python + lxml + defusedxml" echo " Beamer slides: tectonic (optional)" echo "============================================"