mirror of
https://github.com/EvolutionAPI/adk-python.git
synced 2025-07-14 01:41:25 -06:00
189 lines
5.2 KiB
Python
189 lines
5.2 KiB
Python
# Copyright 2025 Google LLC
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
import os
|
|
import shutil
|
|
import subprocess
|
|
from typing import Optional
|
|
|
|
import click
|
|
|
|
_DOCKERFILE_TEMPLATE = """
|
|
FROM python:3.11-slim
|
|
WORKDIR /app
|
|
|
|
# Create a non-root user
|
|
RUN adduser --disabled-password --gecos "" myuser
|
|
|
|
# Change ownership of /app to myuser
|
|
RUN chown -R myuser:myuser /app
|
|
|
|
# Switch to the non-root user
|
|
USER myuser
|
|
|
|
# Set up environment variables - Start
|
|
ENV PATH="/home/myuser/.local/bin:$PATH"
|
|
|
|
ENV GOOGLE_GENAI_USE_VERTEXAI=1
|
|
ENV GOOGLE_CLOUD_PROJECT={gcp_project_id}
|
|
ENV GOOGLE_CLOUD_LOCATION={gcp_region}
|
|
|
|
# Set up environment variables - End
|
|
|
|
# Install ADK - Start
|
|
RUN pip install google-adk
|
|
# Install ADK - End
|
|
|
|
# Copy agent - Start
|
|
|
|
COPY "agents/{app_name}/" "/app/agents/{app_name}/"
|
|
{install_agent_deps}
|
|
|
|
# Copy agent - End
|
|
|
|
EXPOSE {port}
|
|
|
|
CMD adk {command} --port={port} --host=0.0.0.0 {session_db_option} {trace_to_cloud_option} "/app/agents"
|
|
"""
|
|
|
|
|
|
def _resolve_project(project_in_option: Optional[str]) -> str:
|
|
if project_in_option:
|
|
return project_in_option
|
|
|
|
result = subprocess.run(
|
|
['gcloud', 'config', 'get-value', 'project'],
|
|
check=True,
|
|
capture_output=True,
|
|
text=True,
|
|
)
|
|
project = result.stdout.strip()
|
|
click.echo(f'Use default project: {project}')
|
|
return project
|
|
|
|
|
|
def to_cloud_run(
|
|
*,
|
|
agent_folder: str,
|
|
project: Optional[str],
|
|
region: Optional[str],
|
|
service_name: str,
|
|
app_name: str,
|
|
temp_folder: str,
|
|
port: int,
|
|
trace_to_cloud: bool,
|
|
with_ui: bool,
|
|
verbosity: str,
|
|
session_db_url: str,
|
|
):
|
|
"""Deploys an agent to Google Cloud Run.
|
|
|
|
`agent_folder` should contain the following files:
|
|
|
|
- __init__.py
|
|
- agent.py
|
|
- requirements.txt (optional, for additional dependencies)
|
|
- ... (other required source files)
|
|
|
|
The folder structure of temp_folder will be
|
|
|
|
* dist/[google_adk wheel file]
|
|
* agents/[app_name]/
|
|
* agent source code from `agent_folder`
|
|
|
|
Args:
|
|
agent_folder: The folder (absolute path) containing the agent source code.
|
|
project: Google Cloud project id.
|
|
region: Google Cloud region.
|
|
service_name: The service name in Cloud Run.
|
|
app_name: The name of the app, by default, it's basename of `agent_folder`.
|
|
temp_folder: The temp folder for the generated Cloud Run source files.
|
|
port: The port of the ADK api server.
|
|
trace_to_cloud: Whether to enable Cloud Trace.
|
|
with_ui: Whether to deploy with UI.
|
|
verbosity: The verbosity level of the CLI.
|
|
session_db_url: The database URL to connect the session.
|
|
"""
|
|
app_name = app_name or os.path.basename(agent_folder)
|
|
|
|
click.echo(f'Start generating Cloud Run source files in {temp_folder}')
|
|
|
|
# remove temp_folder if exists
|
|
if os.path.exists(temp_folder):
|
|
click.echo('Removing existing files')
|
|
shutil.rmtree(temp_folder)
|
|
|
|
try:
|
|
# copy agent source code
|
|
click.echo('Copying agent source code...')
|
|
agent_src_path = os.path.join(temp_folder, 'agents', app_name)
|
|
shutil.copytree(agent_folder, agent_src_path)
|
|
requirements_txt_path = os.path.join(agent_src_path, 'requirements.txt')
|
|
install_agent_deps = (
|
|
f'RUN pip install -r "/app/agents/{app_name}/requirements.txt"'
|
|
if os.path.exists(requirements_txt_path)
|
|
else ''
|
|
)
|
|
click.echo('Copying agent source code complete.')
|
|
|
|
# create Dockerfile
|
|
click.echo('Creating Dockerfile...')
|
|
dockerfile_content = _DOCKERFILE_TEMPLATE.format(
|
|
gcp_project_id=project,
|
|
gcp_region=region,
|
|
app_name=app_name,
|
|
port=port,
|
|
command='web' if with_ui else 'api_server',
|
|
install_agent_deps=install_agent_deps,
|
|
session_db_option=f'--session_db_url={session_db_url}'
|
|
if session_db_url
|
|
else '',
|
|
trace_to_cloud_option='--trace_to_cloud' if trace_to_cloud else '',
|
|
)
|
|
dockerfile_path = os.path.join(temp_folder, 'Dockerfile')
|
|
os.makedirs(temp_folder, exist_ok=True)
|
|
with open(dockerfile_path, 'w', encoding='utf-8') as f:
|
|
f.write(
|
|
dockerfile_content,
|
|
)
|
|
click.echo(f'Creating Dockerfile complete: {dockerfile_path}')
|
|
|
|
# Deploy to Cloud Run
|
|
click.echo('Deploying to Cloud Run...')
|
|
region_options = ['--region', region] if region else []
|
|
project = _resolve_project(project)
|
|
subprocess.run(
|
|
[
|
|
'gcloud',
|
|
'run',
|
|
'deploy',
|
|
service_name,
|
|
'--source',
|
|
temp_folder,
|
|
'--project',
|
|
project,
|
|
*region_options,
|
|
'--port',
|
|
str(port),
|
|
'--verbosity',
|
|
verbosity,
|
|
'--labels',
|
|
'created-by=adk',
|
|
],
|
|
check=True,
|
|
)
|
|
finally:
|
|
click.echo(f'Cleaning up the temp folder: {temp_folder}')
|
|
shutil.rmtree(temp_folder)
|