mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
feat(installer): add an interactive version chooser
This commit is contained in:
parent
5a816818dc
commit
6a8a3b50bc
@ -34,6 +34,7 @@ class Installer:
|
|||||||
print("A virtual environment is already activated. Please 'deactivate' before installation.")
|
print("A virtual environment is already activated. Please 'deactivate' before installation.")
|
||||||
sys.exit(-1)
|
sys.exit(-1)
|
||||||
self.bootstrap()
|
self.bootstrap()
|
||||||
|
self.available_releases = get_github_releases()
|
||||||
|
|
||||||
def mktemp_venv(self) -> TemporaryDirectory:
|
def mktemp_venv(self) -> TemporaryDirectory:
|
||||||
"""
|
"""
|
||||||
@ -105,7 +106,7 @@ class Installer:
|
|||||||
return venv_dir
|
return venv_dir
|
||||||
|
|
||||||
def install(
|
def install(
|
||||||
self, root: str = "~/invokeai", version: str = "latest", yes_to_all=False, find_links: Optional[Path] = None
|
self, version=None, root: str = "~/invokeai", yes_to_all=False, find_links: Optional[Path] = None
|
||||||
) -> None:
|
) -> None:
|
||||||
"""
|
"""
|
||||||
Install the InvokeAI application into the given runtime path
|
Install the InvokeAI application into the given runtime path
|
||||||
@ -122,7 +123,9 @@ class Installer:
|
|||||||
|
|
||||||
import messages
|
import messages
|
||||||
|
|
||||||
messages.welcome()
|
messages.welcome(self.available_releases)
|
||||||
|
|
||||||
|
version = messages.choose_version(self.available_releases)
|
||||||
|
|
||||||
auto_dest = Path(os.environ.get("INVOKEAI_ROOT", root)).expanduser().resolve()
|
auto_dest = Path(os.environ.get("INVOKEAI_ROOT", root)).expanduser().resolve()
|
||||||
destination = auto_dest if yes_to_all else messages.dest_path(root)
|
destination = auto_dest if yes_to_all else messages.dest_path(root)
|
||||||
@ -157,7 +160,7 @@ class InvokeAiInstance:
|
|||||||
A single runtime directory *may* be shared by multiple virtual environments, though this isn't currently tested or supported.
|
A single runtime directory *may* be shared by multiple virtual environments, though this isn't currently tested or supported.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self, runtime: Path, venv: Path, version: str) -> None:
|
def __init__(self, runtime: Path, venv: Path, version: str = "stable") -> None:
|
||||||
self.runtime = runtime
|
self.runtime = runtime
|
||||||
self.venv = venv
|
self.venv = venv
|
||||||
self.pip = get_pip_from_venv(venv)
|
self.pip = get_pip_from_venv(venv)
|
||||||
@ -179,21 +182,7 @@ class InvokeAiInstance:
|
|||||||
|
|
||||||
def install(self, extra_index_url=None, optional_modules=None, find_links=None):
|
def install(self, extra_index_url=None, optional_modules=None, find_links=None):
|
||||||
"""
|
"""
|
||||||
Install this instance, including dependencies and the app itself
|
Install the package from PyPi.
|
||||||
|
|
||||||
:param extra_index_url: the "--extra-index-url ..." line for pip to look in extra indexes.
|
|
||||||
:type extra_index_url: str
|
|
||||||
"""
|
|
||||||
|
|
||||||
import messages
|
|
||||||
|
|
||||||
messages.simple_banner("Installing the InvokeAI Application :art:")
|
|
||||||
self.install_app(extra_index_url, optional_modules, find_links)
|
|
||||||
|
|
||||||
def install_app(self, extra_index_url=None, optional_modules=None, find_links=None):
|
|
||||||
"""
|
|
||||||
Install the application with pip.
|
|
||||||
Supports installation from PyPi or from a local source directory.
|
|
||||||
|
|
||||||
:param extra_index_url: the "--extra-index-url ..." line for pip to look in extra indexes.
|
:param extra_index_url: the "--extra-index-url ..." line for pip to look in extra indexes.
|
||||||
:type extra_index_url: str
|
:type extra_index_url: str
|
||||||
@ -205,15 +194,26 @@ class InvokeAiInstance:
|
|||||||
:type find_links: Path
|
:type find_links: Path
|
||||||
"""
|
"""
|
||||||
|
|
||||||
## this only applies to pypi installs; TODO actually use this
|
import messages
|
||||||
if self.version == "pre":
|
|
||||||
|
# not currently used, but may be useful for "install most recent version" option
|
||||||
|
if self.version == "prerelease":
|
||||||
version = None
|
version = None
|
||||||
pre = "--pre"
|
pre_flag = "--pre"
|
||||||
|
elif self.version == "stable":
|
||||||
|
version = None
|
||||||
|
pre_flag = None
|
||||||
else:
|
else:
|
||||||
version = self.version
|
version = self.version
|
||||||
pre = None
|
pre_flag = None
|
||||||
|
|
||||||
src = f"invokeai=={version}" if version is not None else "invokeai"
|
src = "invokeai"
|
||||||
|
if optional_modules:
|
||||||
|
src += optional_modules
|
||||||
|
if version:
|
||||||
|
src += f"=={version}"
|
||||||
|
|
||||||
|
messages.simple_banner("Installing the InvokeAI Application :art:")
|
||||||
|
|
||||||
from plumbum import FG, ProcessExecutionError, local # type: ignore
|
from plumbum import FG, ProcessExecutionError, local # type: ignore
|
||||||
|
|
||||||
@ -225,12 +225,12 @@ class InvokeAiInstance:
|
|||||||
"--require-virtualenv",
|
"--require-virtualenv",
|
||||||
"--force-reinstall",
|
"--force-reinstall",
|
||||||
"--use-pep517",
|
"--use-pep517",
|
||||||
str(src) + (optional_modules if optional_modules else ""),
|
str(src),
|
||||||
"--find-links" if find_links is not None else None,
|
"--find-links" if find_links is not None else None,
|
||||||
find_links,
|
find_links,
|
||||||
"--extra-index-url" if extra_index_url is not None else None,
|
"--extra-index-url" if extra_index_url is not None else None,
|
||||||
extra_index_url,
|
extra_index_url,
|
||||||
pre,
|
pre_flag,
|
||||||
]
|
]
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
@ -9,7 +9,7 @@ from enum import Enum
|
|||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
from prompt_toolkit import HTML, prompt
|
from prompt_toolkit import HTML, prompt
|
||||||
from prompt_toolkit.completion import PathCompleter
|
from prompt_toolkit.completion import FuzzyWordCompleter, PathCompleter
|
||||||
from prompt_toolkit.validation import Validator
|
from prompt_toolkit.validation import Validator
|
||||||
from rich import box, print
|
from rich import box, print
|
||||||
from rich.console import Console, Group, group
|
from rich.console import Console, Group, group
|
||||||
@ -62,6 +62,33 @@ def welcome():
|
|||||||
console.line()
|
console.line()
|
||||||
|
|
||||||
|
|
||||||
|
def choose_version(available_releases: tuple | None = None) -> str:
|
||||||
|
"""
|
||||||
|
Prompt the user to choose an Invoke version to install
|
||||||
|
"""
|
||||||
|
|
||||||
|
# short circuit if we couldn't get a version list
|
||||||
|
# still try to install the latest stable version
|
||||||
|
if available_releases is None:
|
||||||
|
return "stable"
|
||||||
|
|
||||||
|
console.print(":grey_question: [orange3]Please choose an Invoke version to install.")
|
||||||
|
|
||||||
|
choices = available_releases[0] + available_releases[1]
|
||||||
|
|
||||||
|
response = prompt(
|
||||||
|
message=f" <Enter> to install the recommended release ({choices[0]}). <Tab> or type to pick a version: ",
|
||||||
|
complete_while_typing=True,
|
||||||
|
completer=FuzzyWordCompleter(choices),
|
||||||
|
)
|
||||||
|
|
||||||
|
console.print(f" Version {choices[0]} will be installed.")
|
||||||
|
console.line()
|
||||||
|
|
||||||
|
return "stable" if response == "" else response
|
||||||
|
|
||||||
|
return response
|
||||||
|
|
||||||
def confirm_install(dest: Path) -> bool:
|
def confirm_install(dest: Path) -> bool:
|
||||||
if dest.exists():
|
if dest.exists():
|
||||||
print(f":exclamation: Directory {dest} already exists :exclamation:")
|
print(f":exclamation: Directory {dest} already exists :exclamation:")
|
||||||
|
Loading…
Reference in New Issue
Block a user