diff --git a/README.md b/README.md index 8f843ed3..fccb3e35 100644 --- a/README.md +++ b/README.md @@ -67,6 +67,11 @@ bash <(curl -fsSL https://openrouter.ai/lab/spawn/sprite/amazonq.sh) ``` For automation or CI/CD, set environment variables: +#### Cline + +```bash +bash <(curl -fsSL https://openrouter.ai/lab/spawn/sprite/cline.sh) +``` #### Claude Code ```bash @@ -164,6 +169,11 @@ OPENROUTER_API_KEY=sk-or-v1-xxxxx \ bash <(curl -fsSL https://openrouter.ai/lab/spawn/hetzner/claude.sh) ``` +#### Cline + +```bash +bash <(curl -fsSL https://openrouter.ai/lab/spawn/hetzner/cline.sh) +``` **Environment Variables:** - `HETZNER_SERVER_NAME` - Name for the server (skips prompt) - `HCLOUD_TOKEN` - Hetzner Cloud API token (skips prompt, saved to `~/.config/spawn/hetzner.json`) @@ -240,6 +250,11 @@ OPENROUTER_API_KEY=sk-or-v1-xxxxx \ bash <(curl -fsSL https://openrouter.ai/lab/spawn/digitalocean/claude.sh) ``` +#### Cline + +```bash +bash <(curl -fsSL https://openrouter.ai/lab/spawn/digitalocean/cline.sh) +``` **Environment Variables:** - `DO_DROPLET_NAME` - Name for the droplet (skips prompt) - `DO_API_TOKEN` - DigitalOcean API token (skips prompt, saved to `~/.config/spawn/digitalocean.json`) @@ -316,6 +331,11 @@ OPENROUTER_API_KEY=sk-or-v1-xxxxx \ bash <(curl -fsSL https://openrouter.ai/lab/spawn/vultr/claude.sh) ``` +#### Cline + +```bash +bash <(curl -fsSL https://openrouter.ai/lab/spawn/vultr/cline.sh) +``` **Environment Variables:** - `VULTR_SERVER_NAME` - Name for the instance (skips prompt) - `VULTR_API_KEY` - Vultr API key (skips prompt, saved to `~/.config/spawn/vultr.json`) @@ -392,6 +412,11 @@ OPENROUTER_API_KEY=sk-or-v1-xxxxx \ bash <(curl -fsSL https://openrouter.ai/lab/spawn/linode/claude.sh) ``` +#### Cline + +```bash +bash <(curl -fsSL https://openrouter.ai/lab/spawn/linode/cline.sh) +``` **Environment Variables:** - `LINODE_SERVER_NAME` - Label for the Linode (skips prompt) - `LINODE_API_TOKEN` - Linode API token (skips prompt, saved to `~/.config/spawn/linode.json`) @@ -467,6 +492,11 @@ OPENROUTER_API_KEY=sk-or-v1-xxxxx \ bash <(curl -fsSL https://openrouter.ai/lab/spawn/aws-lightsail/claude.sh) ``` +#### Cline + +```bash +bash <(curl -fsSL https://openrouter.ai/lab/spawn/aws-lightsail/cline.sh) +``` **Environment Variables:** - `LIGHTSAIL_SERVER_NAME` - Name for the instance (skips prompt) - `OPENROUTER_API_KEY` - Skip OAuth and use this API key directly @@ -536,6 +566,11 @@ bash <(curl -fsSL https://openrouter.ai/lab/spawn/lambda/amazonq.sh) ``` ### Non-Interactive Mode +#### Cline + +```bash +bash <(curl -fsSL https://openrouter.ai/lab/spawn/lambda/cline.sh) +``` ```bash LAMBDA_SERVER_NAME=dev-mk1 \ LAMBDA_API_KEY=your-lambda-api-key \ diff --git a/aws-lightsail/cline.sh b/aws-lightsail/cline.sh new file mode 100755 index 00000000..5f563b81 --- /dev/null +++ b/aws-lightsail/cline.sh @@ -0,0 +1,67 @@ +#!/bin/bash +set -e + +# Source common functions - try local file first, fall back to remote +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" 2>/dev/null && pwd)" +if [[ -f "$SCRIPT_DIR/lib/common.sh" ]]; then + source "$SCRIPT_DIR/lib/common.sh" +else + source <(curl -fsSL https://raw.githubusercontent.com/OpenRouterTeam/spawn/main/aws-lightsail/lib/common.sh) +fi + +log_info "Cline on AWS Lightsail" +echo "" + +# 1. Ensure AWS CLI is configured +ensure_aws_cli + +# 2. Generate + register SSH key +ensure_ssh_key + +# 3. Get instance name and create server +SERVER_NAME=$(get_server_name) +create_server "$SERVER_NAME" + +# 4. Wait for SSH and cloud-init +verify_server_connectivity "$LIGHTSAIL_SERVER_IP" +wait_for_cloud_init "$LIGHTSAIL_SERVER_IP" + +# 5. Install Cline +log_warn "Installing Cline..." +run_server "$LIGHTSAIL_SERVER_IP" "npm install -g cline" +log_info "Cline installed" + +# 6. Get OpenRouter API key +echo "" +if [[ -n "$OPENROUTER_API_KEY" ]]; then + log_info "Using OpenRouter API key from environment" +else + OPENROUTER_API_KEY=$(get_openrouter_api_key_oauth 5180) +fi + +# 7. Inject environment variables into ~/.zshrc +log_warn "Setting up environment variables..." + +ENV_TEMP=$(mktemp) +cat > "$ENV_TEMP" << EOF + +# [spawn:env] +export OPENROUTER_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_BASE_URL="https://openrouter.ai/api/v1" +EOF + +upload_file "$LIGHTSAIL_SERVER_IP" "$ENV_TEMP" "/tmp/env_config" +run_server "$LIGHTSAIL_SERVER_IP" "cat /tmp/env_config >> ~/.zshrc && rm /tmp/env_config" +rm "$ENV_TEMP" + +echo "" +log_info "Lightsail instance setup completed successfully!" +log_info "Instance: $SERVER_NAME (IP: $LIGHTSAIL_SERVER_IP)" +echo "" + +# 8. Start Cline interactively +log_warn "Starting Cline..." +sleep 1 +clear +interactive_session "$LIGHTSAIL_SERVER_IP" "source ~/.zshrc && cline" diff --git a/digitalocean/cline.sh b/digitalocean/cline.sh new file mode 100755 index 00000000..f97fe261 --- /dev/null +++ b/digitalocean/cline.sh @@ -0,0 +1,54 @@ +#!/bin/bash +set -e + +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" 2>/dev/null && pwd)" +if [[ -f "$SCRIPT_DIR/lib/common.sh" ]]; then + source "$SCRIPT_DIR/lib/common.sh" +else + source <(curl -fsSL https://raw.githubusercontent.com/OpenRouterTeam/spawn/main/digitalocean/lib/common.sh) +fi + +log_info "Cline on DigitalOcean" +echo "" + +ensure_do_token +ensure_ssh_key + +DROPLET_NAME=$(get_server_name) +create_server "$DROPLET_NAME" +verify_server_connectivity "$DO_SERVER_IP" +wait_for_cloud_init "$DO_SERVER_IP" + +log_warn "Installing Cline..." +run_server "$DO_SERVER_IP" "npm install -g cline" +log_info "Cline installed" + +echo "" +if [[ -n "$OPENROUTER_API_KEY" ]]; then + log_info "Using OpenRouter API key from environment" +else + OPENROUTER_API_KEY=$(get_openrouter_api_key_oauth 5180) +fi + +log_warn "Setting up environment variables..." +ENV_TEMP=$(mktemp) +cat > "$ENV_TEMP" << EOF + +# [spawn:env] +export OPENROUTER_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_BASE_URL="https://openrouter.ai/api/v1" +EOF +upload_file "$DO_SERVER_IP" "$ENV_TEMP" "/tmp/env_config" +run_server "$DO_SERVER_IP" "cat /tmp/env_config >> ~/.zshrc && rm /tmp/env_config" +rm "$ENV_TEMP" + +echo "" +log_info "DigitalOcean droplet setup completed successfully!" +log_info "Droplet: $DROPLET_NAME (ID: $DO_DROPLET_ID, IP: $DO_SERVER_IP)" +echo "" + +log_warn "Starting Cline..." +sleep 1 +clear +interactive_session "$DO_SERVER_IP" "source ~/.zshrc && cline" diff --git a/hetzner/cline.sh b/hetzner/cline.sh new file mode 100755 index 00000000..48a1e8a2 --- /dev/null +++ b/hetzner/cline.sh @@ -0,0 +1,54 @@ +#!/bin/bash +set -e + +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" 2>/dev/null && pwd)" +if [[ -f "$SCRIPT_DIR/lib/common.sh" ]]; then + source "$SCRIPT_DIR/lib/common.sh" +else + source <(curl -fsSL https://raw.githubusercontent.com/OpenRouterTeam/spawn/main/hetzner/lib/common.sh) +fi + +log_info "Cline on Hetzner Cloud" +echo "" + +ensure_hcloud_token +ensure_ssh_key + +SERVER_NAME=$(get_server_name) +create_server "$SERVER_NAME" +verify_server_connectivity "$HETZNER_SERVER_IP" +wait_for_cloud_init "$HETZNER_SERVER_IP" + +log_warn "Installing Cline..." +run_server "$HETZNER_SERVER_IP" "npm install -g cline" +log_info "Cline installed" + +echo "" +if [[ -n "$OPENROUTER_API_KEY" ]]; then + log_info "Using OpenRouter API key from environment" +else + OPENROUTER_API_KEY=$(get_openrouter_api_key_oauth 5180) +fi + +log_warn "Setting up environment variables..." +ENV_TEMP=$(mktemp) +cat > "$ENV_TEMP" << EOF + +# [spawn:env] +export OPENROUTER_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_BASE_URL="https://openrouter.ai/api/v1" +EOF +upload_file "$HETZNER_SERVER_IP" "$ENV_TEMP" "/tmp/env_config" +run_server "$HETZNER_SERVER_IP" "cat /tmp/env_config >> ~/.zshrc && rm /tmp/env_config" +rm "$ENV_TEMP" + +echo "" +log_info "Hetzner server setup completed successfully!" +log_info "Server: $SERVER_NAME (ID: $HETZNER_SERVER_ID, IP: $HETZNER_SERVER_IP)" +echo "" + +log_warn "Starting Cline..." +sleep 1 +clear +interactive_session "$HETZNER_SERVER_IP" "source ~/.zshrc && cline" diff --git a/lambda/cline.sh b/lambda/cline.sh new file mode 100755 index 00000000..40dec088 --- /dev/null +++ b/lambda/cline.sh @@ -0,0 +1,67 @@ +#!/bin/bash +set -e + +# Source common functions - try local file first, fall back to remote +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" 2>/dev/null && pwd)" +if [[ -f "$SCRIPT_DIR/lib/common.sh" ]]; then + source "$SCRIPT_DIR/lib/common.sh" +else + source <(curl -fsSL https://raw.githubusercontent.com/OpenRouterTeam/spawn/main/lambda/lib/common.sh) +fi + +log_info "Cline on Lambda Cloud" +echo "" + +# 1. Ensure Lambda API key is configured +ensure_lambda_token + +# 2. Generate + register SSH key +ensure_ssh_key + +# 3. Get instance name and create server +SERVER_NAME=$(get_server_name) +create_server "$SERVER_NAME" + +# 4. Wait for SSH and cloud-init +verify_server_connectivity "$LAMBDA_SERVER_IP" +wait_for_cloud_init "$LAMBDA_SERVER_IP" + +# 5. Install Cline +log_warn "Installing Cline..." +run_server "$LAMBDA_SERVER_IP" "npm install -g cline" +log_info "Cline installed" + +# 6. Get OpenRouter API key +echo "" +if [[ -n "$OPENROUTER_API_KEY" ]]; then + log_info "Using OpenRouter API key from environment" +else + OPENROUTER_API_KEY=$(get_openrouter_api_key_oauth 5180) +fi + +# 7. Inject environment variables into ~/.zshrc +log_warn "Setting up environment variables..." + +ENV_TEMP=$(mktemp) +cat > "$ENV_TEMP" << EOF + +# [spawn:env] +export OPENROUTER_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_BASE_URL="https://openrouter.ai/api/v1" +EOF + +upload_file "$LAMBDA_SERVER_IP" "$ENV_TEMP" "/tmp/env_config" +run_server "$LAMBDA_SERVER_IP" "cat /tmp/env_config >> ~/.zshrc && rm /tmp/env_config" +rm "$ENV_TEMP" + +echo "" +log_info "Lambda Cloud instance setup completed successfully!" +log_info "Instance: $SERVER_NAME (IP: $LAMBDA_SERVER_IP)" +echo "" + +# 8. Start Cline interactively +log_warn "Starting Cline..." +sleep 1 +clear +interactive_session "$LAMBDA_SERVER_IP" "source ~/.zshrc && cline" diff --git a/linode/cline.sh b/linode/cline.sh new file mode 100755 index 00000000..53bcdcac --- /dev/null +++ b/linode/cline.sh @@ -0,0 +1,38 @@ +#!/bin/bash +set -e +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" 2>/dev/null && pwd)" +if [[ -f "$SCRIPT_DIR/lib/common.sh" ]]; then source "$SCRIPT_DIR/lib/common.sh" +else source <(curl -fsSL https://raw.githubusercontent.com/OpenRouterTeam/spawn/main/linode/lib/common.sh); fi +log_info "Cline on Linode" +echo "" +ensure_linode_token +ensure_ssh_key +SERVER_NAME=$(get_server_name) +create_server "$SERVER_NAME" +verify_server_connectivity "$LINODE_SERVER_IP" +wait_for_cloud_init "$LINODE_SERVER_IP" +log_warn "Installing Cline..." +run_server "$LINODE_SERVER_IP" "npm install -g cline" +log_info "Cline installed" +echo "" +if [[ -n "$OPENROUTER_API_KEY" ]]; then log_info "Using OpenRouter API key from environment" +else OPENROUTER_API_KEY=$(get_openrouter_api_key_oauth 5180); fi +log_warn "Setting up environment variables..." +ENV_TEMP=$(mktemp) +cat > "$ENV_TEMP" << EOF + +# [spawn:env] +export OPENROUTER_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_BASE_URL="https://openrouter.ai/api/v1" +EOF +upload_file "$LINODE_SERVER_IP" "$ENV_TEMP" "/tmp/env_config" +run_server "$LINODE_SERVER_IP" "cat /tmp/env_config >> ~/.zshrc && rm /tmp/env_config" +rm "$ENV_TEMP" +echo "" +log_info "Linode setup completed successfully!" +echo "" +log_warn "Starting Cline..." +sleep 1 +clear +interactive_session "$LINODE_SERVER_IP" "source ~/.zshrc && cline" diff --git a/manifest.json b/manifest.json index 286adf0e..625cd2b6 100644 --- a/manifest.json +++ b/manifest.json @@ -147,6 +147,19 @@ "OPENROUTER_API_KEY": "${OPENROUTER_API_KEY}" }, "notes": "Works with OpenRouter via OPENAI_BASE_URL override" + }, + "cline": { + "name": "Cline", + "description": "Open-source AI coding agent for the terminal", + "url": "https://github.com/cline/cline", + "install": "npm install -g cline", + "launch": "cline", + "env": { + "OPENAI_API_KEY": "${OPENROUTER_API_KEY}", + "OPENAI_BASE_URL": "https://openrouter.ai/api/v1", + "OPENROUTER_API_KEY": "${OPENROUTER_API_KEY}" + }, + "notes": "Works with OpenRouter via OPENAI_BASE_URL override" } }, "clouds": { @@ -315,6 +328,13 @@ "vultr/amazonq": "implemented", "linode/amazonq": "implemented", "lambda/amazonq": "implemented", - "aws-lightsail/amazonq": "implemented" + "aws-lightsail/amazonq": "implemented", + "sprite/cline": "implemented", + "hetzner/cline": "implemented", + "digitalocean/cline": "implemented", + "vultr/cline": "implemented", + "linode/cline": "implemented", + "lambda/cline": "implemented", + "aws-lightsail/cline": "implemented" } } diff --git a/sprite/cline.sh b/sprite/cline.sh new file mode 100755 index 00000000..ace86db9 --- /dev/null +++ b/sprite/cline.sh @@ -0,0 +1,53 @@ +#!/bin/bash +set -e + +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" 2>/dev/null && pwd)" +if [[ -f "$SCRIPT_DIR/lib/common.sh" ]]; then + source "$SCRIPT_DIR/lib/common.sh" +else + source <(curl -fsSL https://raw.githubusercontent.com/OpenRouterTeam/spawn/main/sprite/lib/common.sh) +fi + +log_info "Cline on Sprite" +echo "" + +ensure_sprite_installed +ensure_sprite_authenticated + +SPRITE_NAME=$(get_sprite_name) +ensure_sprite_exists "$SPRITE_NAME" 5 +verify_sprite_connectivity "$SPRITE_NAME" + +log_warn "Setting up sprite environment..." +setup_shell_environment "$SPRITE_NAME" + +log_warn "Installing Cline..." +run_sprite "$SPRITE_NAME" "npm install -g cline" + +echo "" +if [[ -n "$OPENROUTER_API_KEY" ]]; then + log_info "Using OpenRouter API key from environment" +else + OPENROUTER_API_KEY=$(get_openrouter_api_key_oauth 5180) +fi + +log_warn "Setting up environment variables..." +ENV_TEMP=$(mktemp) +cat > "$ENV_TEMP" << EOF + +# [spawn:env] +export OPENROUTER_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_BASE_URL="https://openrouter.ai/api/v1" +EOF +sprite exec -s "$SPRITE_NAME" -file "$ENV_TEMP:/tmp/env_config" -- bash -c "cat /tmp/env_config >> ~/.zshrc && rm /tmp/env_config" +rm "$ENV_TEMP" + +echo "" +log_info "Sprite setup completed successfully!" +echo "" + +log_warn "Starting Cline..." +sleep 1 +clear +sprite exec -s "$SPRITE_NAME" -tty -- zsh -c "source ~/.zshrc && cline" diff --git a/vultr/cline.sh b/vultr/cline.sh new file mode 100755 index 00000000..137f3a07 --- /dev/null +++ b/vultr/cline.sh @@ -0,0 +1,54 @@ +#!/bin/bash +set -e + +SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" 2>/dev/null && pwd)" +if [[ -f "$SCRIPT_DIR/lib/common.sh" ]]; then + source "$SCRIPT_DIR/lib/common.sh" +else + source <(curl -fsSL https://raw.githubusercontent.com/OpenRouterTeam/spawn/main/vultr/lib/common.sh) +fi + +log_info "Cline on Vultr" +echo "" + +ensure_vultr_token +ensure_ssh_key + +SERVER_NAME=$(get_server_name) +create_server "$SERVER_NAME" +verify_server_connectivity "$VULTR_SERVER_IP" +wait_for_cloud_init "$VULTR_SERVER_IP" + +log_warn "Installing Cline..." +run_server "$VULTR_SERVER_IP" "npm install -g cline" +log_info "Cline installed" + +echo "" +if [[ -n "$OPENROUTER_API_KEY" ]]; then + log_info "Using OpenRouter API key from environment" +else + OPENROUTER_API_KEY=$(get_openrouter_api_key_oauth 5180) +fi + +log_warn "Setting up environment variables..." +ENV_TEMP=$(mktemp) +cat > "$ENV_TEMP" << EOF + +# [spawn:env] +export OPENROUTER_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_API_KEY="${OPENROUTER_API_KEY}" +export OPENAI_BASE_URL="https://openrouter.ai/api/v1" +EOF +upload_file "$VULTR_SERVER_IP" "$ENV_TEMP" "/tmp/env_config" +run_server "$VULTR_SERVER_IP" "cat /tmp/env_config >> ~/.zshrc && rm /tmp/env_config" +rm "$ENV_TEMP" + +echo "" +log_info "Vultr instance setup completed successfully!" +log_info "Server: $SERVER_NAME (ID: $VULTR_SERVER_ID, IP: $VULTR_SERVER_IP)" +echo "" + +log_warn "Starting Cline..." +sleep 1 +clear +interactive_session "$VULTR_SERVER_IP" "source ~/.zshrc && cline"