mirror of
https://github.com/memohai/Memoh.git
synced 2026-04-25 07:00:48 +09:00
244 lines
7.7 KiB
Bash
Executable File
244 lines
7.7 KiB
Bash
Executable File
#!/bin/sh
|
|
set -e
|
|
|
|
GREEN='\033[0;32m'
|
|
YELLOW='\033[1;33m'
|
|
PURPLE='\033[0;35m'
|
|
RED='\033[0;31m'
|
|
NC='\033[0m'
|
|
|
|
GITHUB_REPO="memohai/Memoh"
|
|
REPO="https://github.com/${GITHUB_REPO}.git"
|
|
DIR="Memoh"
|
|
SILENT=false
|
|
|
|
# Parse flags
|
|
for arg in "$@"; do
|
|
case "$arg" in
|
|
-y|--yes) SILENT=true ;;
|
|
esac
|
|
done
|
|
|
|
# Auto-silent if no TTY available
|
|
if [ "$SILENT" = false ] && ! [ -e /dev/tty ]; then
|
|
SILENT=true
|
|
fi
|
|
|
|
echo "${PURPLE}Memoh One-Click Install${NC}"
|
|
|
|
# Check Docker and determine if sudo is needed
|
|
DOCKER="docker"
|
|
if ! command -v docker >/dev/null 2>&1; then
|
|
echo "${RED}Error: Docker is not installed${NC}"
|
|
echo "Install Docker first: https://docs.docker.com/get-docker/"
|
|
exit 1
|
|
fi
|
|
if ! docker info >/dev/null 2>&1; then
|
|
if sudo docker info >/dev/null 2>&1; then
|
|
DOCKER="sudo docker"
|
|
else
|
|
echo "${RED}Error: Cannot connect to Docker daemon${NC}"
|
|
echo "Try: sudo usermod -aG docker \$USER && newgrp docker"
|
|
exit 1
|
|
fi
|
|
fi
|
|
if ! $DOCKER compose version >/dev/null 2>&1; then
|
|
echo "${RED}Error: Docker Compose v2 is required${NC}"
|
|
echo "Install: https://docs.docker.com/compose/install/"
|
|
exit 1
|
|
fi
|
|
echo "${GREEN}✓ Docker and Docker Compose detected${NC}"
|
|
|
|
# Resolve version: use MEMOH_VERSION env if set, otherwise fetch latest release
|
|
VERSION_USER_PROVIDED=false
|
|
if [ -n "$MEMOH_VERSION" ]; then
|
|
VERSION_USER_PROVIDED=true
|
|
echo "${GREEN}✓ Using specified version: ${MEMOH_VERSION}${NC}"
|
|
else
|
|
fetch_latest_version() {
|
|
if command -v curl >/dev/null 2>&1; then
|
|
curl -fsSL "https://api.github.com/repos/${GITHUB_REPO}/releases/latest" 2>/dev/null
|
|
elif command -v wget >/dev/null 2>&1; then
|
|
wget -qO- "https://api.github.com/repos/${GITHUB_REPO}/releases/latest" 2>/dev/null
|
|
else
|
|
echo "${RED}Error: curl or wget is required${NC}" >&2
|
|
exit 1
|
|
fi
|
|
}
|
|
MEMOH_VERSION=$(fetch_latest_version | grep '"tag_name"' | sed 's/.*"tag_name": *"\([^"]*\)".*/\1/')
|
|
if [ -n "$MEMOH_VERSION" ]; then
|
|
echo "${GREEN}✓ Latest release: ${MEMOH_VERSION}${NC}"
|
|
else
|
|
echo "${YELLOW}Warning: Failed to fetch latest release tag, falling back to main branch${NC}"
|
|
fi
|
|
fi
|
|
|
|
# Docker image tag: pin to version only when user explicitly specified MEMOH_VERSION
|
|
if [ "$VERSION_USER_PROVIDED" = true ]; then
|
|
MEMOH_DOCKER_VERSION=$(echo "$MEMOH_VERSION" | sed 's/^v//')
|
|
else
|
|
MEMOH_DOCKER_VERSION="latest"
|
|
fi
|
|
echo "${GREEN}✓ Docker image version: ${MEMOH_DOCKER_VERSION}${NC}"
|
|
|
|
# Generate random JWT secret
|
|
gen_secret() {
|
|
if command -v openssl >/dev/null 2>&1; then
|
|
openssl rand -base64 32
|
|
else
|
|
head -c 32 /dev/urandom | base64 | tr -d '\n'
|
|
fi
|
|
}
|
|
|
|
# Configuration defaults (expand ~ for paths)
|
|
WORKSPACE_DEFAULT="${HOME:-/tmp}/memoh"
|
|
MEMOH_DATA_DIR_DEFAULT="${HOME:-/tmp}/memoh/data"
|
|
ADMIN_USER="admin"
|
|
ADMIN_PASS="admin123"
|
|
JWT_SECRET="$(gen_secret)"
|
|
PG_PASS="memoh123"
|
|
WORKSPACE="$WORKSPACE_DEFAULT"
|
|
MEMOH_DATA_DIR="$MEMOH_DATA_DIR_DEFAULT"
|
|
USE_CN_MIRROR=false
|
|
|
|
if [ "$SILENT" = false ]; then
|
|
echo "Configure Memoh (press Enter to use defaults):" > /dev/tty
|
|
echo "" > /dev/tty
|
|
|
|
printf " Workspace (install and clone here) [%s]: " "~/memoh" > /dev/tty
|
|
read -r input < /dev/tty || true
|
|
if [ -n "$input" ]; then
|
|
case "$input" in
|
|
~) WORKSPACE="${HOME:-/tmp}" ;;
|
|
~/*) WORKSPACE="${HOME:-/tmp}${input#\~}" ;;
|
|
*) WORKSPACE="$input" ;;
|
|
esac
|
|
fi
|
|
|
|
printf " Data directory (bind mount for server container data) [%s]: " "$WORKSPACE/data" > /dev/tty
|
|
read -r input < /dev/tty || true
|
|
if [ -n "$input" ]; then
|
|
case "$input" in
|
|
~) MEMOH_DATA_DIR="${HOME:-/tmp}" ;;
|
|
~/*) MEMOH_DATA_DIR="${HOME:-/tmp}${input#\~}" ;;
|
|
*) MEMOH_DATA_DIR="$input" ;;
|
|
esac
|
|
else
|
|
MEMOH_DATA_DIR="$WORKSPACE/data"
|
|
fi
|
|
|
|
printf " Admin username [%s]: " "$ADMIN_USER" > /dev/tty
|
|
read -r input < /dev/tty || true
|
|
[ -n "$input" ] && ADMIN_USER="$input"
|
|
|
|
printf " Admin password [%s]: " "$ADMIN_PASS" > /dev/tty
|
|
read -r input < /dev/tty || true
|
|
[ -n "$input" ] && ADMIN_PASS="$input"
|
|
|
|
printf " JWT secret [auto-generated]: " > /dev/tty
|
|
read -r input < /dev/tty || true
|
|
[ -n "$input" ] && JWT_SECRET="$input"
|
|
|
|
printf " Postgres password [%s]: " "$PG_PASS" > /dev/tty
|
|
read -r input < /dev/tty || true
|
|
[ -n "$input" ] && PG_PASS="$input"
|
|
|
|
printf " Use China mainland mirror? (y/N): " > /dev/tty
|
|
read -r input < /dev/tty || true
|
|
case "$input" in
|
|
[Yy]|[Yy][Ee][Ss]) USE_CN_MIRROR=true ;;
|
|
esac
|
|
|
|
echo "" > /dev/tty
|
|
fi
|
|
|
|
# Enter workspace (all operations run here)
|
|
mkdir -p "$WORKSPACE"
|
|
cd "$WORKSPACE"
|
|
|
|
# Clone or update
|
|
if [ -d "$DIR" ]; then
|
|
echo "Updating existing installation in $WORKSPACE..."
|
|
cd "$DIR"
|
|
if [ -n "$MEMOH_VERSION" ]; then
|
|
git fetch --depth 1 origin tag "$MEMOH_VERSION"
|
|
git checkout "$MEMOH_VERSION"
|
|
else
|
|
git fetch --depth 1 origin main
|
|
git checkout main 2>/dev/null || git checkout -b main --track origin/main
|
|
git reset --hard origin/main
|
|
fi
|
|
else
|
|
echo "Cloning Memoh into $WORKSPACE..."
|
|
if [ -n "$MEMOH_VERSION" ]; then
|
|
git clone --depth 1 --branch "$MEMOH_VERSION" "$REPO" "$DIR"
|
|
else
|
|
git clone --depth 1 "$REPO" "$DIR"
|
|
fi
|
|
cd "$DIR"
|
|
fi
|
|
|
|
# Pin Docker image versions in docker-compose.yml
|
|
if [ "$MEMOH_DOCKER_VERSION" != "latest" ]; then
|
|
sed -i.bak "s|memohai/server:latest|memohai/server:${MEMOH_DOCKER_VERSION}|g" docker-compose.yml
|
|
sed -i.bak "s|memohai/agent:latest|memohai/agent:${MEMOH_DOCKER_VERSION}|g" docker-compose.yml
|
|
sed -i.bak "s|memohai/web:latest|memohai/web:${MEMOH_DOCKER_VERSION}|g" docker-compose.yml
|
|
sed -i.bak "s|memohai/browser:latest|memohai/browser:${MEMOH_DOCKER_VERSION}|g" docker-compose.yml
|
|
rm -f docker-compose.yml.bak
|
|
echo "${GREEN}✓ Docker images pinned to ${MEMOH_DOCKER_VERSION}${NC}"
|
|
fi
|
|
|
|
# Generate config.toml from template
|
|
cp conf/app.docker.toml config.toml
|
|
sed -i.bak "s|username = \"admin\"|username = \"${ADMIN_USER}\"|" config.toml
|
|
sed -i.bak "s|password = \"admin123\"|password = \"${ADMIN_PASS}\"|" config.toml
|
|
sed -i.bak "s|jwt_secret = \".*\"|jwt_secret = \"${JWT_SECRET}\"|" config.toml
|
|
sed -i.bak "s|password = \"memoh123\"|password = \"${PG_PASS}\"|" config.toml
|
|
export POSTGRES_PASSWORD="${PG_PASS}"
|
|
if [ "$USE_CN_MIRROR" = true ]; then
|
|
sed -i.bak 's|# registry = "memoh.cn"|registry = "memoh.cn"|' config.toml
|
|
fi
|
|
rm -f config.toml.bak
|
|
|
|
# Use generated config and data dir
|
|
INSTALL_DIR="$(pwd)"
|
|
export MEMOH_CONFIG=./config.toml
|
|
export MEMOH_DATA_DIR
|
|
mkdir -p "$MEMOH_DATA_DIR"
|
|
|
|
COMPOSE_FILES="-f docker-compose.yml"
|
|
if [ "$USE_CN_MIRROR" = true ]; then
|
|
COMPOSE_FILES="$COMPOSE_FILES -f docker/docker-compose.cn.yml"
|
|
echo "${GREEN}✓ Using China mainland mirror (memoh.cn)${NC}"
|
|
fi
|
|
|
|
echo POSTGRES_PASSWORD="${PG_PASS}" >> .env
|
|
echo MEMOH_CONFIG=./config.toml >> .env
|
|
echo MEMOH_DATA_DIR="{$MEMOH_DATA_DIR}" >> .env
|
|
|
|
echo ""
|
|
echo "${GREEN}Pulling latest Docker images...${NC}"
|
|
$DOCKER compose $COMPOSE_FILES pull
|
|
|
|
echo ""
|
|
echo "${GREEN}Starting services (first startup may take a few minutes)...${NC}"
|
|
$DOCKER compose $COMPOSE_FILES up -d
|
|
|
|
echo ""
|
|
echo "${GREEN}✅ Memoh is running!${NC}${NC}"
|
|
echo ""
|
|
echo " 🌐 Web UI: http://localhost:8082"
|
|
echo " 🔌 API: http://localhost:8080"
|
|
echo " 🤖 Agent Gateway: http://localhost:8081"
|
|
echo " 🌍 Browser Gateway: http://localhost:8083"
|
|
echo ""
|
|
echo " 🔑 Admin login: ${ADMIN_USER} / ${ADMIN_PASS}"
|
|
echo ""
|
|
COMPOSE_CMD="$DOCKER compose $COMPOSE_FILES"
|
|
echo "📋 Commands:"
|
|
echo " cd ${INSTALL_DIR} && ${COMPOSE_CMD} ps # Status"
|
|
echo " cd ${INSTALL_DIR} && ${COMPOSE_CMD} logs -f # Logs"
|
|
echo " cd ${INSTALL_DIR} && ${COMPOSE_CMD} down # Stop"
|
|
echo ""
|
|
echo "${YELLOW}⏳ First startup may take 1-2 minutes, please be patient.${NC}"
|