-
Notifications
You must be signed in to change notification settings - Fork 16
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
4 changed files
with
289 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,64 @@ | ||
name: MOR Agents Build Linux | ||
|
||
on: | ||
push: | ||
branches: [ main ] | ||
pull_request: | ||
branches: [ main ] | ||
workflow_dispatch: | ||
|
||
jobs: | ||
build: | ||
runs-on: ubuntu-latest | ||
|
||
steps: | ||
- uses: actions/checkout@v4 | ||
with: | ||
submodules: 'recursive' | ||
|
||
- name: Set up Python | ||
uses: actions/setup-python@v5 | ||
with: | ||
python-version: '3.12' | ||
|
||
- name: Install dependencies | ||
run: | | ||
python -m pip install --upgrade pip | ||
pip install -r requirements.txt | ||
pip install pyinstaller | ||
- name: Build with PyInstaller | ||
run: | | ||
pyinstaller --name="MORagents" --add-data "images/moragents.png:images" main.py | ||
- name: Create Debian package | ||
run: | | ||
mkdir -p debian/DEBIAN | ||
mkdir -p debian/usr/bin | ||
mkdir -p debian/usr/share/applications | ||
mkdir -p debian/usr/share/icons/hicolor/256x256/apps | ||
cp -r dist/MORagents/* debian/usr/bin/ | ||
cp images/moragents.png debian/usr/share/icons/hicolor/256x256/apps/moragents.png | ||
echo "[Desktop Entry] | ||
Name=MORagents | ||
Exec=/usr/bin/MORagents | ||
Icon=moragents | ||
Type=Application | ||
Categories=Utility;" > debian/usr/share/applications/moragents.desktop | ||
echo "Package: moragents | ||
Version: 1.0 | ||
Section: utils | ||
Priority: optional | ||
Architecture: amd64 | ||
Maintainer: Your Name <[email protected]> | ||
Description: MORagents application | ||
MORagents is a desktop application for AI agents." > debian/DEBIAN/control | ||
cp build_assets/linux/install_moragents.sh debian/usr/bin/ | ||
chmod +x debian/usr/bin/install_moragents.sh | ||
dpkg-deb --build debian moragents.deb | ||
- name: Upload Debian Package | ||
uses: actions/upload-artifact@v4 | ||
with: | ||
name: MORagentsSetup-Linux | ||
path: moragents.deb |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,58 @@ | ||
#!/bin/bash | ||
|
||
# Colors for output | ||
RED='\033[0;31m' | ||
GREEN='\033[0;32m' | ||
NC='\033[0m' # No Color | ||
|
||
# Function to check if a command exists | ||
command_exists() { | ||
command -v "$1" >/dev/null 2>&1 | ||
} | ||
|
||
# Check if script is run as root | ||
if [ "$EUID" -ne 0 ]; then | ||
echo -e "${RED}Please run as root${NC}" | ||
exit 1 | ||
fi | ||
|
||
echo "Welcome to the MORagents installer for Linux" | ||
|
||
# Check and install Docker if not present | ||
if ! command_exists docker; then | ||
echo "Docker not found. Installing Docker..." | ||
curl -fsSL https://get.docker.com -o get-docker.sh | ||
sh get-docker.sh | ||
usermod -aG docker $SUDO_USER | ||
systemctl enable docker | ||
systemctl start docker | ||
else | ||
echo "Docker is already installed." | ||
fi | ||
|
||
# Check and install Python if not present | ||
if ! command_exists python3; then | ||
echo "Python3 not found. Installing Python3..." | ||
apt update | ||
apt install -y python3 python3-pip | ||
else | ||
echo "Python3 is already installed." | ||
fi | ||
|
||
# Install the .deb package | ||
echo "Installing MORagents..." | ||
dpkg -i moragents.deb | ||
apt-get install -f -y | ||
|
||
# Pull necessary Docker images | ||
echo "Pulling Docker images..." | ||
docker pull morpheusai/nginx-agent:latest | ||
docker pull morpheusai/agents:latest | ||
|
||
# Start Docker containers | ||
echo "Starting Docker containers..." | ||
docker run -d --name agents -p 8080:5000 --restart always -v /var/lib/agents -v /app/src morpheusai/agents:latest | ||
docker run -d --name nginx -p 3333:80 morpheusai/nginx-agent:latest | ||
|
||
echo -e "${GREEN}Installation complete!${NC}" | ||
echo "You can now start MORagents from your application menu or by running 'MORagents' in the terminal." |
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,167 @@ | ||
import os | ||
import shutil | ||
import subprocess | ||
|
||
from utils.logger_config import setup_logger | ||
from config import AgentDockerConfig, AgentDockerConfigDeprecate | ||
|
||
logger = setup_logger(__name__) | ||
|
||
def get_docker_path(): | ||
docker_paths = [ | ||
'/usr/bin/docker', # Common Linux path | ||
'/usr/local/bin/docker', # Alternative Linux path | ||
shutil.which('docker') | ||
] | ||
for docker_path in docker_paths: | ||
if docker_path and os.path.exists(docker_path): | ||
return docker_path | ||
|
||
logger.error("Docker executable not found in PATH.") | ||
return None | ||
|
||
def check_docker_installed(docker_path): | ||
try: | ||
subprocess.run([docker_path, "--version"], | ||
check=True, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL) | ||
logger.info(f"Docker was found at {docker_path}") | ||
return True | ||
except (subprocess.CalledProcessError, TypeError) as e: | ||
logger.error(f"Error checking Docker installation: {str(e)}") | ||
return False | ||
|
||
def delete_docker_image(docker_path, image_name): | ||
try: | ||
# List all images | ||
list_command = [docker_path, "images", "--format", "{{.Repository}}:{{.Tag}}"] | ||
output = subprocess.check_output(list_command, universal_newlines=True) | ||
images = output.strip().split("\n") | ||
|
||
# Find the image with the specified name | ||
if image_name in images: | ||
# Remove the image | ||
remove_command = [docker_path, "rmi", "-f", image_name] | ||
subprocess.run(remove_command, check=True) | ||
logger.info(f"Image '{image_name}' deleted successfully.") | ||
else: | ||
pass | ||
|
||
except subprocess.CalledProcessError as e: | ||
logger.warning(f"Error deleting image: {e}") | ||
|
||
def list_containers_for_image(docker_path, image_name): | ||
try: | ||
output = subprocess.check_output( | ||
[docker_path, "ps", "-a", "--filter", f"ancestor={image_name}", "--format", "{{.ID}}"]) | ||
containers = output.decode().strip().split("\n") | ||
return [container for container in containers if container] | ||
except subprocess.CalledProcessError as e: | ||
logger.error(f"Failed to list containers for image '{image_name}': {e}") | ||
return [] | ||
|
||
def remove_container(docker_path, container): | ||
try: | ||
subprocess.run([docker_path, "rm", "-f", container], check=True, stdout=subprocess.DEVNULL, | ||
stderr=subprocess.DEVNULL) | ||
except subprocess.CalledProcessError as e: | ||
logger.error(f"Failed to remove container '{container}': {e}") | ||
|
||
def docker_image_present_on_host(docker_path, image_name): | ||
try: | ||
subprocess.run([docker_path, "inspect", image_name], check=True, stdout=subprocess.DEVNULL, | ||
stderr=subprocess.DEVNULL) | ||
return True | ||
except (subprocess.CalledProcessError, TypeError) as e: | ||
return False | ||
|
||
def remove_containers_for_image(docker_path, image_name): | ||
containers = list_containers_for_image(docker_path, image_name) | ||
for container in containers: | ||
remove_container(docker_path, container) | ||
logger.info(f"Removed container '{container}' for image '{image_name}'") | ||
|
||
def remove_containers_by_name(docker_path, container_name): | ||
try: | ||
list_command = [docker_path, "ps", "-a", "--format", "{{.Names}}"] | ||
output = subprocess.check_output(list_command, universal_newlines=True) | ||
containers = output.strip().split("\n") | ||
|
||
if container_name in containers: | ||
remove_command = [docker_path, "rm", "-f", container_name] | ||
subprocess.run(remove_command, check=True) | ||
logger.info(f"Removed container '{container_name}'") | ||
else: | ||
logger.info(f"Container '{container_name}' not found") | ||
except subprocess.CalledProcessError as e: | ||
logger.error(f"Error removing container '{container_name}': {str(e)}") | ||
|
||
def migration_remove_old_images(docker_path): | ||
for image_name in AgentDockerConfigDeprecate.OLD_IMAGE_NAMES: | ||
if docker_image_present_on_host(docker_path, image_name): | ||
delete_docker_image(docker_path, image_name) | ||
logger.info(f"Deleted image '{image_name} from previous release") | ||
|
||
def pull_docker_images(docker_path): | ||
for image in AgentDockerConfig.get_current_image_names(): | ||
try: | ||
subprocess.run([docker_path, "pull", image], check=True) | ||
logger.info(f"Successfully pulled image: {image}") | ||
except subprocess.CalledProcessError as e: | ||
logger.error(f"Failed to pull image {image}: {e}") | ||
raise | ||
|
||
def start_ollama_server(): | ||
ollama_path = '/usr/local/bin/ollama' # This path might need to be adjusted for Linux | ||
|
||
try: | ||
# Start Ollama server | ||
logger.info("Starting Ollama server...") | ||
subprocess.Popen([ollama_path, "serve"], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL) | ||
logger.info("Ollama server started successfully.") | ||
except Exception as e: | ||
logger.info("Failed to start Ollama server.") | ||
logger.error(f"Failed to start Ollama server: {e}") | ||
|
||
def docker_setup(): | ||
docker_path = get_docker_path() | ||
logger.info(f"Docker path: {docker_path}") | ||
|
||
if not check_docker_installed(docker_path): | ||
logger.critical("Docker is not installed.") | ||
raise RuntimeError("Docker is not installed.") | ||
|
||
# Remove old images and containers | ||
logger.info("Checking whether old images need removal.") | ||
migration_remove_old_images(docker_path) | ||
|
||
for image_name in AgentDockerConfig.get_current_image_names(): | ||
remove_containers_for_image(docker_path, image_name) | ||
|
||
remove_containers_by_name(docker_path, "agents") | ||
remove_containers_by_name(docker_path, "nginx") | ||
|
||
# Pull the latest images | ||
pull_docker_images(docker_path) | ||
|
||
# Spin up Agent container | ||
subprocess.run([ | ||
docker_path, "run", "-d", "--name", "agents", | ||
"-p", "8080:5000", "--restart", "always", | ||
"-v", "/var/lib/agents:/var/lib/agents", "-v", "/app/src:/app/src", # Adjusted volume paths for Linux | ||
AgentDockerConfig.get_current_image_names()[1] # agents image | ||
], check=True) | ||
|
||
# Spin up Nginx container | ||
subprocess.run([ | ||
docker_path, "run", "-d", "--name", "nginx", "-p", "3333:80", | ||
AgentDockerConfig.get_current_image_names()[0] # nginx image | ||
], check=True) | ||
|
||
def main(): | ||
# main() called every time the app is opened (from main.py). Put all app open code here. | ||
logger.info("Starting app...") | ||
start_ollama_server() | ||
docker_setup() | ||
|
||
if __name__ == "__main__": | ||
main() |