"""Manage LangChain apps.""" import shutil import subprocess import sys import warnings from pathlib import Path from typing import Annotated import typer import uvicorn from langchain_cli.utils.events import create_events from langchain_cli.utils.git import ( DependencySource, copy_repo, parse_dependencies, update_repo, ) from langchain_cli.utils.packages import ( LangServeExport, get_langserve_export, get_package_root, ) from langchain_cli.utils.pyproject import ( add_dependencies_to_pyproject_toml, remove_dependencies_from_pyproject_toml, ) REPO_DIR = Path(typer.get_app_dir("langchain")) / "git_repos" app_cli = typer.Typer(no_args_is_help=True, add_completion=False) @app_cli.command() def new( name: Annotated[ str | None, typer.Argument( help="The name of the folder to create", ), ] = None, *, package: Annotated[ list[str] | None, typer.Option(help="Packages to seed the project with"), ] = None, pip: Annotated[ bool | None, typer.Option( "--pip/--no-pip", help="Pip install the template(s) as editable dependencies", ), ] = None, noninteractive: Annotated[ bool, typer.Option( "--non-interactive/--interactive", help="Don't prompt for any input", ), ] = False, ) -> None: """Create a new LangServe application.""" has_packages = package is not None and len(package) > 0 if noninteractive: if name is None: msg = "name is required when --non-interactive is set" raise typer.BadParameter(msg) name_str = name pip_bool = bool(pip) # None should be false else: name_str = name or typer.prompt("What folder would you like to create?") if not has_packages: package = [] package_prompt = "What package would you like to add? (leave blank to skip)" while True: package_str = typer.prompt( package_prompt, default="", show_default=False, ) if not package_str: break package.append(package_str) package_prompt = ( f"{len(package)} added. Any more packages (leave blank to end)?" ) has_packages = len(package) > 0 pip_bool = False if pip is None and has_packages: pip_bool = typer.confirm( "Would you like to install these templates into your environment " "with pip?", default=False, ) # copy over template from ../project_template project_template_dir = Path(__file__).parents[1] / "project_template" destination_dir = Path.cwd() / name_str if name_str != "." else Path.cwd() app_name = name_str if name_str != "." else Path.cwd().name shutil.copytree(project_template_dir, destination_dir, dirs_exist_ok=name == ".") readme = destination_dir / "README.md" readme_contents = readme.read_text() readme.write_text(readme_contents.replace("__app_name__", app_name)) pyproject = destination_dir / "pyproject.toml" pyproject_contents = pyproject.read_text() pyproject.write_text(pyproject_contents.replace("__app_name__", app_name)) # add packages if specified if has_packages: add(package, project_dir=destination_dir, pip=pip_bool) typer.echo(f'\n\nSuccess! Created a new LangChain app under "./{app_name}"!\n\n') typer.echo("Next, enter your new app directory by running:\n") typer.echo(f" cd ./{app_name}\n") typer.echo("Then add templates with commands like:\n") typer.echo(" langchain app add extraction-openai-functions") typer.echo( " langchain app add git+ssh://git@github.com/efriis/simple-pirate.git\n\n", ) @app_cli.command() def add( dependencies: Annotated[ list[str] | None, typer.Argument(help="The dependency to add"), ] = None, *, api_path: Annotated[ list[str] | None, typer.Option(help="API paths to add"), ] = None, project_dir: Annotated[ Path | None, typer.Option(help="The project directory"), ] = None, repo: Annotated[ list[str] | None, typer.Option(help="Install templates from a specific github repo instead"), ] = None, branch: Annotated[ list[str] | None, typer.Option(help="Install templates from a specific branch"), ] = None, pip: Annotated[ bool, typer.Option( "--pip/--no-pip", help="Pip install the template(s) as editable dependencies", prompt="Would you like to `pip install -e` the template(s)?", ), ], ) -> None: """Add the specified template to the current LangServe app. e.g.: `langchain app add extraction-openai-functions` `langchain app add git+ssh://git@github.com/efriis/simple-pirate.git` """ if branch is None: branch = [] if repo is None: repo = [] if api_path is None: api_path = [] if not branch and not repo: warnings.warn( "Adding templates from the default branch and repo is deprecated." " At a minimum, you will have to add `--branch v0.2` for this to work", stacklevel=2, ) parsed_deps = parse_dependencies(dependencies, repo, branch, api_path) project_root = get_package_root(project_dir) package_dir = project_root / "packages" create_events( [{"event": "serve add", "properties": {"parsed_dep": d}} for d in parsed_deps], ) # group by repo/ref grouped: dict[tuple[str, str | None], list[DependencySource]] = {} for dep in parsed_deps: key_tup = (dep["git"], dep["ref"]) lst = grouped.get(key_tup, []) lst.append(dep) grouped[key_tup] = lst installed_destination_paths: list[Path] = [] installed_destination_names: list[str] = [] installed_exports: list[LangServeExport] = [] for (git, ref), group_deps in grouped.items(): if len(group_deps) == 1: typer.echo(f"Adding {git}@{ref}...") else: typer.echo(f"Adding {len(group_deps)} templates from {git}@{ref}") source_repo_path = update_repo(git, ref, REPO_DIR) for dep in group_deps: source_path = ( source_repo_path / dep["subdirectory"] if dep["subdirectory"] else source_repo_path ) pyproject_path = source_path / "pyproject.toml" if not pyproject_path.exists(): typer.echo(f"Could not find {pyproject_path}") continue langserve_export = get_langserve_export(pyproject_path) # default path to package_name inner_api_path = dep["api_path"] or langserve_export["package_name"] destination_path = package_dir / inner_api_path if destination_path.exists(): typer.echo( f"Folder {inner_api_path} already exists. Skipping...", ) continue copy_repo(source_path, destination_path) typer.echo(f" - Downloaded {dep['subdirectory']} to {inner_api_path}") installed_destination_paths.append(destination_path) installed_destination_names.append(inner_api_path) installed_exports.append(langserve_export) if len(installed_destination_paths) == 0: typer.echo("No packages installed. Exiting.") return try: add_dependencies_to_pyproject_toml( project_root / "pyproject.toml", zip(installed_destination_names, installed_destination_paths, strict=False), ) except Exception: # Can fail if user modified/removed pyproject.toml typer.echo("Failed to add dependencies to pyproject.toml, continuing...") try: cwd = Path.cwd() installed_destination_strs = [ str(p.relative_to(cwd)) for p in installed_destination_paths ] except ValueError: # Can fail if the cwd is not a parent of the package typer.echo("Failed to print install command, continuing...") else: if pip: cmd = ["pip", "install", "-e", *installed_destination_strs] cmd_str = " \\\n ".join(installed_destination_strs) typer.echo(f"Running: pip install -e \\\n {cmd_str}") subprocess.run(cmd, cwd=cwd, check=True) # noqa: S603 chain_names = [] for e in installed_exports: original_candidate = f"{e['package_name'].replace('-', '_')}_chain" candidate = original_candidate i = 2 while candidate in chain_names: candidate = original_candidate + "_" + str(i) i += 1 chain_names.append(candidate) api_paths = [ str(Path("/") / path.relative_to(package_dir)) for path in installed_destination_paths ] imports = [ f"from {e['module']} import {e['attr']} as {name}" for e, name in zip(installed_exports, chain_names, strict=False) ] routes = [ f'add_routes(app, {name}, path="{path}")' for name, path in zip(chain_names, api_paths, strict=False) ] t = ( "this template" if len(chain_names) == 1 else f"these {len(chain_names)} templates" ) lines = [ "", f"To use {t}, add the following to your app:\n\n```", "", *imports, "", *routes, "```", ] typer.echo("\n".join(lines)) @app_cli.command() def remove( api_paths: Annotated[list[str], typer.Argument(help="The API paths to remove")], *, project_dir: Annotated[ Path | None, typer.Option(help="The project directory"), ] = None, ) -> None: """Remove the specified package from the current LangServe app.""" project_root = get_package_root(project_dir) project_pyproject = project_root / "pyproject.toml" package_root = project_root / "packages" remove_deps: list[str] = [] for api_path in api_paths: package_dir = package_root / api_path if not package_dir.exists(): typer.echo(f"Package {api_path} does not exist. Skipping...") continue try: pyproject = package_dir / "pyproject.toml" langserve_export = get_langserve_export(pyproject) typer.echo(f"Removing {langserve_export['package_name']}...") shutil.rmtree(package_dir) remove_deps.append(api_path) except OSError as exc: typer.echo(f"Failed to remove {api_path}: {exc}") try: remove_dependencies_from_pyproject_toml(project_pyproject, remove_deps) except Exception: # Can fail if user modified/removed pyproject.toml typer.echo("Failed to remove dependencies from pyproject.toml.") @app_cli.command() def serve( *, port: Annotated[ int | None, typer.Option(help="The port to run the server on"), ] = None, host: Annotated[ str | None, typer.Option(help="The host to run the server on"), ] = None, app: Annotated[ str | None, typer.Option(help="The app to run, e.g. `app.server:app`"), ] = None, ) -> None: """Start the LangServe app.""" # add current dir as first entry of path sys.path.append(str(Path.cwd())) app_str = app if app is not None else "app.server:app" host_str = host if host is not None else "127.0.0.1" uvicorn.run( app_str, host=host_str, port=port if port is not None else 8000, reload=True, )