From 2801b35e5722e446aaee558d11324448a86a2d82 Mon Sep 17 00:00:00 2001 From: "Jonathan C. McKinney" Date: Fri, 2 Jun 2023 02:30:57 -0700 Subject: [PATCH] Add setup.py wheel building option --- README.md | 19 ++++ docs/README_LangChain.md | 5 +- reqs_optional/requirements_optional_faiss.txt | 2 - .../requirements_optional_faiss_cpu.txt | 1 + setup.py | 99 +++++++++++++++++++ 5 files changed, 123 insertions(+), 3 deletions(-) create mode 100644 reqs_optional/requirements_optional_faiss_cpu.txt create mode 100644 setup.py diff --git a/README.md b/README.md index 280410948..2734e8f54 100644 --- a/README.md +++ b/README.md @@ -20,6 +20,7 @@ Turn ★ into ⭐ (top-right corner) if you like the project! * [CLI chat](#cli-chat) * [Gradio UI](#gradio-ui) * [Client API](#client-api) + * [Python Wheel](#python-wheel) * [Development](#development) * [Help](#help) * [LangChain file types supported](docs/README_LangChain.md#supported-datatypes) @@ -313,6 +314,24 @@ We disable background uploads by disabling telemetry for huggingface, gradio, an `generate.py` by default runs a gradio server, which also gives access to client API using gradio client. See example [test code](client_test.py) or other tests in our [tests](https://github.com/h2oai/h2ogpt/blob/main/tests/test_client_calls.py). Any element in [gradio_runner.py](gradio_runner.py) with `api_name` defined can be accessed via the gradio client. +#### Python Wheel + +The wheel adds all dependencies except specialized dependencies like 4-bit and flash-attention. For GPU build do: +```bash +GPU=1 python setup.py sdist bdist_wheel +pip install dist/h2ogpt-*-py3-none-any.whl --extra-index-url https://download.pytorch.org/whl/cu117 +``` +replace `*` with actual version built if more than one. Or for CPU mode do: +```bash +GPU=0 python setup.py sdist bdist_wheel +pip install dist/h2ogpt-*-py3-none-any.whl +``` +or copy the `whl` file and make fresh env with `python=3.10` and install it there. + +The wheel is not required to use h2oGPT locally from repo, but makes it portable with all required dependencies. + +See [setup.py](setup.py) for controlling other options via environment variables. + ### Development - To create a development environment for training and generation, follow the [installation instructions](docs/INSTALL.md). diff --git a/docs/README_LangChain.md b/docs/README_LangChain.md index 5ef6093c0..ce938429b 100644 --- a/docs/README_LangChain.md +++ b/docs/README_LangChain.md @@ -10,7 +10,10 @@ To support GPU FAISS database, run: ```bash pip install -r reqs_optional/requirements_optional_faiss.txt ``` -or if you have no GPUs, you can still use FAISS but comment-out the faiss-gpu line and uncomment the faiss-cpu line. +or for CPU FAISS database, run: +```bash +pip install -r reqs_optional/requirements_optional_faiss_cpu.txt +``` ## Supported Datatypes diff --git a/reqs_optional/requirements_optional_faiss.txt b/reqs_optional/requirements_optional_faiss.txt index 2d6db8df2..f749cfd94 100644 --- a/reqs_optional/requirements_optional_faiss.txt +++ b/reqs_optional/requirements_optional_faiss.txt @@ -1,3 +1 @@ -# choose: -#faiss-cpu faiss-gpu==1.7.2 diff --git a/reqs_optional/requirements_optional_faiss_cpu.txt b/reqs_optional/requirements_optional_faiss_cpu.txt new file mode 100644 index 000000000..ebcb33946 --- /dev/null +++ b/reqs_optional/requirements_optional_faiss_cpu.txt @@ -0,0 +1 @@ +faiss-cpu diff --git a/setup.py b/setup.py new file mode 100644 index 000000000..1a6cda50d --- /dev/null +++ b/setup.py @@ -0,0 +1,99 @@ +import os +import glob +import setuptools +from typing import List + +from utils import get_ngpus_vis + + +def parse_requirements(file_name: str) -> List[str]: + with open(file_name) as f: + required = f.read().splitlines() + required = [x for x in required if not x.strip().startswith("#")] + required = [x if 'git+http' not in x else 'peft @' + x for x in required] + required = [x for x in required if x] + return required + + +do_install_optional_req = bool(int(os.environ.get('OPTIONAL', '1'))) +do_install_gpl = bool(int(os.environ.get('GPL', '1'))) +do_install_extra_training = bool(int(os.environ.get('TRAINING', '0'))) +do_install_wiki_extra = bool(int(os.environ.get('WIKI_EXTRA', '0'))) +# avoid 4bit deps as part of package until part of normal deps on pypi +do_install_4bit = bool(int(os.environ.get('4BIT', '0'))) +do_install_flash = bool(int(os.environ.get('FLASH', '0'))) +have_gpus = int(get_ngpus_vis(raise_if_exception=False) > 0) +do_gpu = bool(int(os.environ.get('GPU', str(have_gpus)))) + +# base requirements list +base_req = 'requirements.txt' +install_requires = parse_requirements(base_req) +# list of optional requirement files +all_optional_reqs = glob.glob('reqs_optional/requirements*.txt') +exceptional_deps = [] +if not do_install_4bit: + exceptional_deps.append('reqs_optional/requirements_optional_4bit.txt') +if not do_install_flash: + exceptional_deps.append('reqs_optional/requirements_optional_flashattention.txt') +if not do_install_gpl: + exceptional_deps.append('reqs_optional/requirements_optional_langchain.gpllike.txt') +if do_gpu: + exceptional_deps.append('reqs_optional/requirements_optional_faiss_cpu.txt') +if not do_gpu: + exceptional_deps.append('reqs_optional/requirements_optional_faiss.txt') +if not do_install_extra_training: + exceptional_deps.append('reqs_optional/requirements_optional_training.txt') +if not do_install_wiki_extra: + exceptional_deps.append('reqs_optional/requirements_optional_wikiprocessing.txt') + +all_optional_reqs = [x for x in all_optional_reqs if x not in exceptional_deps] + +if do_install_optional_req: + for opt_req in all_optional_reqs: + if opt_req == base_req: + continue + install_requires.extend(parse_requirements(opt_req)) + +# User-friendly description from README.md +current_directory = os.path.dirname(os.path.abspath(__file__)) +try: + with open(os.path.join(current_directory, 'README.md'), encoding='utf-8') as f: + long_description = f.read() +except Exception: + long_description = '' + +setuptools.setup( + # Name of the package + name='h2ogpt', + # Packages to include into the distribution + packages=['h2ogpt'], + #package_dir={'h2ogpt': 'src/h2ogpt'}, + package_dir={'h2ogpt': './'}, + # Start with a small number and increase it with + # every change you make https://semver.org + version='0.1.0', + # Chose a license from here: https: // + # help.github.com / articles / licensing - a - + # repository. For example: MIT + license='https://opensource.org/license/apache-2-0/', + # Short description of your library + description='', + # Long description of your library + long_description=long_description, + long_description_content_type='text/markdown', + # Your name + author='H2O.ai', + # Your email + author_email='jon.mckinney@h2o.ai, arno@h2o.ai', + # Either the link to your github or to your website + url='https://github.com/h2oai/h2ogpt', + # Link from which the project can be downloaded + download_url='', + # List of keywords + keywords=['LLM', 'AI'], + # List of packages to install with this one + install_requires=install_requires, + # https://pypi.org/classifiers/ + classifiers=[], + python_requires=">=3.10", +)