zilongpa commited on
Commit
55be9e4
·
1 Parent(s): dc1b9b1

Upload folder using huggingface_hub

Browse files
.env ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ MODEL_PATH = "/path-to/Llama-2-7b-chat-hf"
2
+ LOAD_IN_8BIT = True
3
+ LOAD_IN_4BIT = False
4
+ LLAMA_CPP = False
5
+
6
+ MAX_MAX_NEW_TOKENS = 2048
7
+ DEFAULT_MAX_NEW_TOKENS = 1024
8
+ MAX_INPUT_TOKEN_LENGTH = 4000
9
+
10
+ DEFAULT_SYSTEM_PROMPT = "You are a helpful, respectful and honest assistant. Always answer as helpfully as possible, while being safe. Your answers should not include any harmful, unethical, racist, sexist, toxic, dangerous, or illegal content. Please ensure that your responses are socially unbiased and positive in nature. If a question does not make any sense, or is not factually coherent, explain why instead of answering something not correct. If you don't know the answer to a question, please don't share false information."
.gitattributes CHANGED
@@ -33,3 +33,4 @@ saved_model/**/* filter=lfs diff=lfs merge=lfs -text
33
  *.zip filter=lfs diff=lfs merge=lfs -text
34
  *.zst filter=lfs diff=lfs merge=lfs -text
35
  *tfevents* filter=lfs diff=lfs merge=lfs -text
 
 
33
  *.zip filter=lfs diff=lfs merge=lfs -text
34
  *.zst filter=lfs diff=lfs merge=lfs -text
35
  *tfevents* filter=lfs diff=lfs merge=lfs -text
36
+ 76437bc4e8bea417641aaa076508098a7158e664c1cecfabfa41df497a27f98c filter=lfs diff=lfs merge=lfs -text
.gitignore ADDED
@@ -0,0 +1,8 @@
 
 
 
 
 
 
 
 
 
1
+ models
2
+ dist
3
+
4
+ .DS_Store
5
+ .vscode
6
+
7
+ __pycache__
8
+ gradio_cached_examples
76437bc4e8bea417641aaa076508098a7158e664c1cecfabfa41df497a27f98c ADDED
@@ -0,0 +1,3 @@
 
 
 
 
1
+ version https://git-lfs.github.com/spec/v1
2
+ oid sha256:76437bc4e8bea417641aaa076508098a7158e664c1cecfabfa41df497a27f98c
3
+ size 3825517184
CONTRIBUTING.md ADDED
@@ -0,0 +1,90 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # Contributing to [llama2-webui](https://github.com/liltom-eth/llama2-webui)
2
+
3
+ We love your input! We want to make contributing to this project as easy and transparent as possible, whether it's:
4
+
5
+ - Reporting a bug
6
+ - Proposing new features
7
+ - Discussing the current state of the code
8
+ - Update README.md
9
+ - Submitting a PR
10
+
11
+ ## Using GitHub's [issues](https://github.com/liltom-eth/llama2-webui/issues)
12
+
13
+ We use GitHub issues to track public bugs. Report a bug by [opening a new issue](https://github.com/liltom-eth/llama2-webui/issues). It's that easy!
14
+
15
+ Thanks for **[jlb1504](https://github.com/jlb1504)** for reporting the [first issue](https://github.com/liltom-eth/llama2-webui/issues/1)!
16
+
17
+ **Great Bug Reports** tend to have:
18
+
19
+ - A quick summary and/or background
20
+ - Steps to reproduce
21
+ - Be specific!
22
+ - Give a sample code if you can.
23
+ - What you expected would happen
24
+ - What actually happens
25
+ - Notes (possibly including why you think this might be happening, or stuff you tried that didn't work)
26
+
27
+ Proposing new features are also welcome.
28
+
29
+ ## Pull Request
30
+
31
+ All pull requests are welcome. For example, you update the `README.md` to help users to better understand the usage.
32
+
33
+ ### Clone the repository
34
+
35
+ 1. Create a user account on GitHub if you do not already have one.
36
+
37
+ 2. Fork the project [repository](https://github.com/liltom-eth/llama2-webui): click on the *Fork* button near the top of the page. This creates a copy of the code under your account on GitHub.
38
+
39
+ 3. Clone this copy to your local disk:
40
+
41
+ ```
42
+ git clone git@github.com:liltom-eth/llama2-webui.git
43
+ cd llama2-webui
44
+ ```
45
+
46
+ ### Implement your changes
47
+
48
+ 1. Create a branch to hold your changes:
49
+
50
+ ```
51
+ git checkout -b my-feature
52
+ ```
53
+
54
+ and start making changes. Never work on the main branch!
55
+
56
+ 2. Start your work on this branch.
57
+
58
+ 3. When you’re done editing, do:
59
+
60
+ ```
61
+ git add <MODIFIED FILES>
62
+ git commit
63
+ ```
64
+
65
+ to record your changes in [git](https://git-scm.com/).
66
+
67
+ ### Submit your contribution
68
+
69
+ 1. If everything works fine, push your local branch to the remote server with:
70
+
71
+ ```
72
+ git push -u origin my-feature
73
+ ```
74
+
75
+ 2. Go to the web page of your fork and click "Create pull request" to send your changes for review.
76
+
77
+ ```{todo}
78
+ Find more detailed information in [creating a PR]. You might also want to open
79
+ the PR as a draft first and mark it as ready for review after the feedbacks
80
+ from the continuous integration (CI) system or any required fixes.
81
+ ```
82
+
83
+ ## License
84
+
85
+ By contributing, you agree that your contributions will be licensed under its MIT License.
86
+
87
+ ## Questions?
88
+
89
+ Email us at [liltom.eth@gmail.com](mailto:liltom.eth@gmail.com)
90
+
LICENSE ADDED
@@ -0,0 +1,21 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ MIT License
2
+
3
+ Copyright (c) 2023 Tom
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
README.md CHANGED
@@ -1,12 +1,217 @@
1
  ---
2
- title: Llama2 Webui
3
- emoji: 📈
4
- colorFrom: yellow
5
- colorTo: purple
6
  sdk: gradio
7
- sdk_version: 3.39.0
8
- app_file: app.py
9
- pinned: false
10
  ---
 
11
 
12
- Check out the configuration reference at https://huggingface.co/docs/hub/spaces-config-reference
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
  ---
2
+ title: llama2-webui
3
+ app_file: app_4bit_ggml.py
 
 
4
  sdk: gradio
5
+ sdk_version: 3.37.0
 
 
6
  ---
7
+ # llama2-webui
8
 
9
+ Running Llama 2 with gradio web UI on GPU or CPU from anywhere (Linux/Windows/Mac).
10
+ - Supporting all Llama 2 models (7B, 13B, 70B, GPTQ, GGML) with 8-bit, 4-bit mode.
11
+ - Supporting GPU inference with at least 6 GB VRAM, and CPU inference.
12
+
13
+ ![screenshot](./static/screenshot.png)
14
+
15
+ ## Features
16
+
17
+ - Supporting models: [Llama-2-7b](https://huggingface.co/meta-llama/Llama-2-7b-chat-hf)/[13b](https://huggingface.co/llamaste/Llama-2-13b-chat-hf)/[70b](https://huggingface.co/llamaste/Llama-2-70b-chat-hf), all [Llama-2-GPTQ](https://huggingface.co/TheBloke/Llama-2-7b-Chat-GPTQ), all [Llama-2-GGML](https://huggingface.co/TheBloke/Llama-2-7B-Chat-GGML) ...
18
+ - Supporting model backends
19
+ - Nvidia GPU: tranformers, [bitsandbytes(8-bit inference)](https://github.com/TimDettmers/bitsandbytes), [AutoGPTQ(4-bit inference)](https://github.com/PanQiWei/AutoGPTQ)
20
+ - GPU inference with at least 6 GB VRAM
21
+
22
+ - CPU, Mac/AMD GPU: [llama.cpp](https://github.com/ggerganov/llama.cpp)
23
+ - CPU inference [Demo](https://twitter.com/liltom_eth/status/1682791729207070720?s=20) on Macbook Air.
24
+
25
+ - Web UI interface: gradio
26
+
27
+ ## Contents
28
+
29
+ - [Install](#install)
30
+ - [Download Llama-2 Models](#download-llama-2-models)
31
+ - [Model List](#model-list)
32
+ - [Download Script](#download-script)
33
+ - [Usage](#usage)
34
+ - [Config Examples](#config-examples)
35
+ - [Start Web UI](#start-web-ui)
36
+ - [Run on Nvidia GPU](#run-on-nvidia-gpu)
37
+ - [Run on Low Memory GPU with 8 bit](#run-on-low-memory-gpu-with-8-bit)
38
+ - [Run on Low Memory GPU with 4 bit](#run-on-low-memory-gpu-with-4-bit)
39
+ - [Run on CPU](#run-on-cpu)
40
+ - [Mac GPU and AMD/Nvidia GPU Acceleration](#mac-gpu-and-amdnvidia-gpu-acceleration)
41
+ - [Benchmark](#benchmark)
42
+ - [Contributing](#contributing)
43
+ - [License](#license)
44
+
45
+
46
+
47
+ ## Install
48
+ ### Method 1: From [PyPI](https://pypi.org/project/llama2-wrapper/)
49
+ ```
50
+ pip install llama2-wrapper
51
+ ```
52
+ ### Method 2: From Source:
53
+ ```
54
+ git clone https://github.com/liltom-eth/llama2-webui.git
55
+ cd llama2-webui
56
+ pip install -r requirements.txt
57
+ ```
58
+ ### Install Issues:
59
+ `bitsandbytes >= 0.39` may not work on older NVIDIA GPUs. In that case, to use `LOAD_IN_8BIT`, you may have to downgrade like this:
60
+
61
+ - `pip install bitsandbytes==0.38.1`
62
+
63
+ `bitsandbytes` also need a special install for Windows:
64
+ ```
65
+ pip uninstall bitsandbytes
66
+ pip install https://github.com/jllllll/bitsandbytes-windows-webui/releases/download/wheels/bitsandbytes-0.41.0-py3-none-win_amd64.whl
67
+ ```
68
+
69
+ ## Download Llama-2 Models
70
+
71
+ Llama 2 is a collection of pre-trained and fine-tuned generative text models ranging in scale from 7 billion to 70 billion parameters.
72
+
73
+ Llama-2-7b-Chat-GPTQ is the GPTQ model files for [Meta's Llama 2 7b Chat](https://huggingface.co/meta-llama/Llama-2-7b-chat-hf). GPTQ 4-bit Llama-2 model require less GPU VRAM to run it.
74
+
75
+ ### Model List
76
+
77
+ | Model Name | set MODEL_PATH in .env | Download URL |
78
+ | ------------------------------ | ---------------------------------------- | ------------------------------------------------------------ |
79
+ | meta-llama/Llama-2-7b-chat-hf | /path-to/Llama-2-7b-chat-hf | [Link](https://huggingface.co/llamaste/Llama-2-7b-chat-hf) |
80
+ | meta-llama/Llama-2-13b-chat-hf | /path-to/Llama-2-13b-chat-hf | [Link](https://huggingface.co/llamaste/Llama-2-13b-chat-hf) |
81
+ | meta-llama/Llama-2-70b-chat-hf | /path-to/Llama-2-70b-chat-hf | [Link](https://huggingface.co/llamaste/Llama-2-70b-chat-hf) |
82
+ | meta-llama/Llama-2-7b-hf | /path-to/Llama-2-7b-hf | [Link](https://huggingface.co/meta-llama/Llama-2-7b-hf) |
83
+ | meta-llama/Llama-2-13b-hf | /path-to/Llama-2-13b-hf | [Link](https://huggingface.co/meta-llama/Llama-2-13b-hf) |
84
+ | meta-llama/Llama-2-70b-hf | /path-to/Llama-2-70b-hf | [Link](https://huggingface.co/meta-llama/Llama-2-70b-hf) |
85
+ | TheBloke/Llama-2-7b-Chat-GPTQ | /path-to/Llama-2-7b-Chat-GPTQ | [Link](https://huggingface.co/TheBloke/Llama-2-7b-Chat-GPTQ) |
86
+ | TheBloke/Llama-2-7B-Chat-GGML | /path-to/llama-2-7b-chat.ggmlv3.q4_0.bin | [Link](https://huggingface.co/TheBloke/Llama-2-7B-Chat-GGML) |
87
+ | ... | ... | ... |
88
+
89
+ Running 4-bit model `Llama-2-7b-Chat-GPTQ` needs GPU with 6GB VRAM.
90
+
91
+ Running 4-bit model `llama-2-7b-chat.ggmlv3.q4_0.bin` needs CPU with 6GB RAM. There is also a list of other 2, 3, 4, 5, 6, 8-bit GGML models that can be used from [TheBloke/Llama-2-7B-Chat-GGML](https://huggingface.co/TheBloke/Llama-2-7B-Chat-GGML).
92
+
93
+ ### Download Script
94
+
95
+ These models can be downloaded from the link using CMD like:
96
+
97
+ ```bash
98
+ # Make sure you have git-lfs installed (https://git-lfs.com)
99
+ git lfs install
100
+ git clone git@hf.co:meta-llama/Llama-2-7b-chat-hf
101
+ ```
102
+
103
+ To download Llama 2 models, you need to request access from [https://ai.meta.com/llama/](https://ai.meta.com/llama/) and also enable access on repos like [meta-llama/Llama-2-7b-chat-hf](https://huggingface.co/meta-llama/Llama-2-7b-chat-hf/tree/main). Requests will be processed in hours.
104
+
105
+ For GPTQ models like [TheBloke/Llama-2-7b-Chat-GPTQ](https://huggingface.co/TheBloke/Llama-2-7b-Chat-GPTQ), you can directly download without requesting access.
106
+
107
+ For GGML models like [TheBloke/Llama-2-7B-Chat-GGML](https://huggingface.co/TheBloke/Llama-2-7B-Chat-GGML), you can directly download without requesting access.
108
+
109
+ ## Usage
110
+
111
+ ### Config Examples
112
+
113
+ Setup your `MODEL_PATH` and model configs in `.env` file.
114
+
115
+ There are some examples in `./env_examples/` folder.
116
+
117
+ | Model Setup | Example .env |
118
+ | --------------------------------- | --------------------------- |
119
+ | Llama-2-7b-chat-hf 8-bit on GPU | .env.7b_8bit_example |
120
+ | Llama-2-7b-Chat-GPTQ 4-bit on GPU | .env.7b_gptq_example |
121
+ | Llama-2-7B-Chat-GGML 4bit on CPU | .env.7b_ggmlv3_q4_0_example |
122
+ | Llama-2-13b-chat-hf on GPU | .env.13b_example |
123
+ | ... | ... |
124
+
125
+ ### Start Web UI
126
+
127
+ Run chatbot with web UI:
128
+
129
+ ```
130
+ python app.py
131
+ ```
132
+
133
+ ### Run on Nvidia GPU
134
+
135
+ The running requires around 14GB of GPU VRAM for Llama-2-7b and 28GB of GPU VRAM for Llama-2-13b.
136
+
137
+ If you are running on multiple GPUs, the model will be loaded automatically on GPUs and split the VRAM usage. That allows you to run Llama-2-7b (requires 14GB of GPU VRAM) on a setup like 2 GPUs (11GB VRAM each).
138
+
139
+ #### Run on Low Memory GPU with 8 bit
140
+
141
+ If you do not have enough memory, you can set up your `LOAD_IN_8BIT` as `True` in `.env`. This can reduce memory usage by around half with slightly degraded model quality. It is compatible with the CPU, GPU, and Metal backend.
142
+
143
+ Llama-2-7b with 8-bit compression can run on a single GPU with 8 GB of VRAM, like an Nvidia RTX 2080Ti, RTX 4080, T4, V100 (16GB).
144
+
145
+ #### Run on Low Memory GPU with 4 bit
146
+
147
+ If you want to run 4 bit Llama-2 model like `Llama-2-7b-Chat-GPTQ`, you can set up your `LOAD_IN_4BIT` as `True` in `.env` like example `.env.7b_gptq_example`.
148
+
149
+ Make sure you have downloaded the 4-bit model from `Llama-2-7b-Chat-GPTQ` and set the `MODEL_PATH` and arguments in `.env` file.
150
+
151
+ `Llama-2-7b-Chat-GPTQ` can run on a single GPU with 6 GB of VRAM.
152
+
153
+ ### Run on CPU
154
+
155
+ Run Llama-2 model on CPU requires [llama.cpp](https://github.com/ggerganov/llama.cpp) dependency and [llama.cpp Python Bindings](https://github.com/abetlen/llama-cpp-python), which are already installed.
156
+
157
+
158
+ Download GGML models like `llama-2-7b-chat.ggmlv3.q4_0.bin` following [Download Llama-2 Models](#download-llama-2-models) section. `llama-2-7b-chat.ggmlv3.q4_0.bin` model requires at least 6 GB RAM to run on CPU.
159
+
160
+ Set up configs like `.env.7b_ggmlv3_q4_0_example` from `env_examples` as `.env`.
161
+
162
+ Run web UI `python app.py` .
163
+
164
+
165
+
166
+ #### Mac GPU and AMD/Nvidia GPU Acceleration
167
+
168
+ If you would like to use Mac GPU and AMD/Nvidia GPU for acceleration, check these:
169
+
170
+ - [Installation with OpenBLAS / cuBLAS / CLBlast / Metal](https://github.com/abetlen/llama-cpp-python#installation-with-openblas--cublas--clblast--metal)
171
+
172
+ - [MacOS Install with Metal GPU](https://github.com/abetlen/llama-cpp-python/blob/main/docs/install/macos.md)
173
+
174
+ ### Benchmark
175
+
176
+ Run benchmark script to compute performance on your device:
177
+
178
+ ```bash
179
+ python benchmark.py
180
+ ```
181
+
182
+ `benchmark.py` will load the same `.env` as `app.py`.
183
+
184
+ Some benchmark performance:
185
+
186
+ | Model | Precision | Device | GPU VRAM | Speed (tokens / sec) | load time (s) |
187
+ | -------------------- | --------- | ------------------ | ----------- | -------------------- | ------------- |
188
+ | Llama-2-7b-chat-hf | 8bit | NVIDIA RTX 2080 Ti | 7.7 GB VRAM | 3.76 | 783.87 |
189
+ | Llama-2-7b-Chat-GPTQ | 4 bit | NVIDIA RTX 2080 Ti | 5.8 GB VRAM | 12.08 | 192.91 |
190
+ | Llama-2-7B-Chat-GGML | 4 bit | Intel i7-8700 | 5.1GB RAM | 4.16 | 105.75 |
191
+
192
+ Check / contribute the performance of your device in the full [performance doc](./docs/performance.md).
193
+
194
+ ## Contributing
195
+
196
+ Kindly read our [Contributing Guide](CONTRIBUTING.md) to learn and understand about our development process.
197
+
198
+ ### All Contributors
199
+
200
+ <a href="https://github.com/liltom-eth/llama2-webui/graphs/contributors">
201
+ <img src="https://contrib.rocks/image?repo=liltom-eth/llama2-webui" />
202
+ </a>
203
+
204
+ ## License
205
+
206
+ MIT - see [MIT License](LICENSE)
207
+
208
+ This project enables users to adapt it freely for proprietary purposes without any restrictions.
209
+
210
+ ## Credits
211
+
212
+ - https://huggingface.co/meta-llama/Llama-2-7b-chat-hf
213
+ - https://huggingface.co/spaces/huggingface-projects/llama-2-7b-chat
214
+ - https://huggingface.co/TheBloke/Llama-2-7b-Chat-GPTQ
215
+ - [https://github.com/ggerganov/llama.cpp](https://github.com/ggerganov/llama.cpp)
216
+ - [https://github.com/TimDettmers/bitsandbytes](https://github.com/TimDettmers/bitsandbytes)
217
+ - [https://github.com/PanQiWei/AutoGPTQ](https://github.com/PanQiWei/AutoGPTQ)
app.py ADDED
@@ -0,0 +1,322 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import os
2
+ from typing import Iterator
3
+
4
+ import gradio as gr
5
+
6
+ from dotenv import load_dotenv
7
+ from distutils.util import strtobool
8
+
9
+ from llama2_wrapper import LLAMA2_WRAPPER
10
+
11
+ load_dotenv()
12
+
13
+ DEFAULT_SYSTEM_PROMPT = (
14
+ os.getenv("DEFAULT_SYSTEM_PROMPT")
15
+ if os.getenv("DEFAULT_SYSTEM_PROMPT") is not None
16
+ else ""
17
+ )
18
+ MAX_MAX_NEW_TOKENS = (
19
+ int(os.getenv("MAX_MAX_NEW_TOKENS"))
20
+ if os.getenv("DEFAULT_MAX_NEW_TOKENS") is not None
21
+ else 2048
22
+ )
23
+ DEFAULT_MAX_NEW_TOKENS = (
24
+ int(os.getenv("DEFAULT_MAX_NEW_TOKENS"))
25
+ if os.getenv("DEFAULT_MAX_NEW_TOKENS") is not None
26
+ else 1024
27
+ )
28
+ MAX_INPUT_TOKEN_LENGTH = (
29
+ int(os.getenv("MAX_INPUT_TOKEN_LENGTH"))
30
+ if os.getenv("MAX_INPUT_TOKEN_LENGTH") is not None
31
+ else 4000
32
+ )
33
+
34
+ MODEL_PATH = os.getenv("MODEL_PATH")
35
+ assert MODEL_PATH is not None, f"MODEL_PATH is required, got: {MODEL_PATH}"
36
+
37
+ LOAD_IN_8BIT = bool(strtobool(os.getenv("LOAD_IN_8BIT", "True")))
38
+
39
+ LOAD_IN_4BIT = bool(strtobool(os.getenv("LOAD_IN_4BIT", "True")))
40
+
41
+ LLAMA_CPP = bool(strtobool(os.getenv("LLAMA_CPP", "True")))
42
+
43
+ if LLAMA_CPP:
44
+ print("Running on CPU with llama.cpp.")
45
+ else:
46
+ import torch
47
+
48
+ if torch.cuda.is_available():
49
+ print("Running on GPU with torch transformers.")
50
+ else:
51
+ print("CUDA not found.")
52
+
53
+ config = {
54
+ "model_name": MODEL_PATH,
55
+ "load_in_8bit": LOAD_IN_8BIT,
56
+ "load_in_4bit": LOAD_IN_4BIT,
57
+ "llama_cpp": LLAMA_CPP,
58
+ "MAX_INPUT_TOKEN_LENGTH": MAX_INPUT_TOKEN_LENGTH,
59
+ }
60
+ llama2_wrapper = LLAMA2_WRAPPER(config)
61
+ llama2_wrapper.init_tokenizer()
62
+ llama2_wrapper.init_model()
63
+
64
+ DESCRIPTION = """
65
+ # llama2-webui
66
+
67
+ This is a chatbot based on Llama-2.
68
+ - Supporting models: [Llama-2-7b](https://huggingface.co/TheBloke/Llama-2-7B-Chat-GGML)/[13b](https://huggingface.co/llamaste/Llama-2-13b-chat-hf)/[70b](https://huggingface.co/llamaste/Llama-2-70b-chat-hf), all [Llama-2-GPTQ](https://huggingface.co/TheBloke/Llama-2-7b-Chat-GPTQ), all [Llama-2-GGML](https://huggingface.co/TheBloke/Llama-2-7B-Chat-GGML) ...
69
+ - Supporting model backends
70
+ - Nvidia GPU(at least 6 GB VRAM): tranformers, [bitsandbytes(8-bit inference)](https://github.com/TimDettmers/bitsandbytes), [AutoGPTQ(4-bit inference)](https://github.com/PanQiWei/AutoGPTQ)
71
+ - CPU(at least 6 GB RAM), Mac/AMD GPU: [llama.cpp](https://github.com/ggerganov/llama.cpp)
72
+ """
73
+
74
+
75
+ def clear_and_save_textbox(message: str) -> tuple[str, str]:
76
+ return "", message
77
+
78
+
79
+ def display_input(
80
+ message: str, history: list[tuple[str, str]]
81
+ ) -> list[tuple[str, str]]:
82
+ history.append((message, ""))
83
+ return history
84
+
85
+
86
+ def delete_prev_fn(history: list[tuple[str, str]]) -> tuple[list[tuple[str, str]], str]:
87
+ try:
88
+ message, _ = history.pop()
89
+ except IndexError:
90
+ message = ""
91
+ return history, message or ""
92
+
93
+
94
+ def generate(
95
+ message: str,
96
+ history_with_input: list[tuple[str, str]],
97
+ system_prompt: str,
98
+ max_new_tokens: int,
99
+ temperature: float,
100
+ top_p: float,
101
+ top_k: int,
102
+ ) -> Iterator[list[tuple[str, str]]]:
103
+ if max_new_tokens > MAX_MAX_NEW_TOKENS:
104
+ raise ValueError
105
+
106
+ history = history_with_input[:-1]
107
+ generator = llama2_wrapper.run(
108
+ message, history, system_prompt, max_new_tokens, temperature, top_p, top_k
109
+ )
110
+ try:
111
+ first_response = next(generator)
112
+ yield history + [(message, first_response)]
113
+ except StopIteration:
114
+ yield history + [(message, "")]
115
+ for response in generator:
116
+ yield history + [(message, response)]
117
+
118
+
119
+ def process_example(message: str) -> tuple[str, list[tuple[str, str]]]:
120
+ generator = generate(message, [], DEFAULT_SYSTEM_PROMPT, 1024, 1, 0.95, 50)
121
+ for x in generator:
122
+ pass
123
+ return "", x
124
+
125
+
126
+ def check_input_token_length(
127
+ message: str, chat_history: list[tuple[str, str]], system_prompt: str
128
+ ) -> None:
129
+ input_token_length = llama2_wrapper.get_input_token_length(
130
+ message, chat_history, system_prompt
131
+ )
132
+ if input_token_length > MAX_INPUT_TOKEN_LENGTH:
133
+ raise gr.Error(
134
+ f"The accumulated input is too long ({input_token_length} > {MAX_INPUT_TOKEN_LENGTH}). Clear your chat history and try again."
135
+ )
136
+
137
+
138
+ with gr.Blocks(css="style.css") as demo:
139
+ gr.Markdown(DESCRIPTION)
140
+
141
+ with gr.Group():
142
+ chatbot = gr.Chatbot(label="Chatbot")
143
+ with gr.Row():
144
+ textbox = gr.Textbox(
145
+ container=False,
146
+ show_label=False,
147
+ placeholder="Type a message...",
148
+ scale=10,
149
+ )
150
+ submit_button = gr.Button("Submit", variant="primary", scale=1, min_width=0)
151
+ with gr.Row():
152
+ retry_button = gr.Button("🔄 Retry", variant="secondary")
153
+ undo_button = gr.Button("↩️ Undo", variant="secondary")
154
+ clear_button = gr.Button("🗑️ Clear", variant="secondary")
155
+
156
+ saved_input = gr.State()
157
+
158
+ with gr.Accordion(label="Advanced options", open=False):
159
+ system_prompt = gr.Textbox(
160
+ label="System prompt", value=DEFAULT_SYSTEM_PROMPT, lines=6
161
+ )
162
+ max_new_tokens = gr.Slider(
163
+ label="Max new tokens",
164
+ minimum=1,
165
+ maximum=MAX_MAX_NEW_TOKENS,
166
+ step=1,
167
+ value=DEFAULT_MAX_NEW_TOKENS,
168
+ )
169
+ temperature = gr.Slider(
170
+ label="Temperature",
171
+ minimum=0.1,
172
+ maximum=4.0,
173
+ step=0.1,
174
+ value=1.0,
175
+ )
176
+ top_p = gr.Slider(
177
+ label="Top-p (nucleus sampling)",
178
+ minimum=0.05,
179
+ maximum=1.0,
180
+ step=0.05,
181
+ value=0.95,
182
+ )
183
+ top_k = gr.Slider(
184
+ label="Top-k",
185
+ minimum=1,
186
+ maximum=1000,
187
+ step=1,
188
+ value=50,
189
+ )
190
+
191
+ gr.Examples(
192
+ examples=[
193
+ "Hello there! How are you doing?",
194
+ "Can you explain briefly to me what is the Python programming language?",
195
+ "Explain the plot of Cinderella in a sentence.",
196
+ "How many hours does it take a man to eat a Helicopter?",
197
+ "Write a 100-word article on 'Benefits of Open-Source in AI research'",
198
+ ],
199
+ inputs=textbox,
200
+ outputs=[textbox, chatbot],
201
+ fn=process_example,
202
+ cache_examples=True,
203
+ )
204
+
205
+ textbox.submit(
206
+ fn=clear_and_save_textbox,
207
+ inputs=textbox,
208
+ outputs=[textbox, saved_input],
209
+ api_name=False,
210
+ queue=False,
211
+ ).then(
212
+ fn=display_input,
213
+ inputs=[saved_input, chatbot],
214
+ outputs=chatbot,
215
+ api_name=False,
216
+ queue=False,
217
+ ).then(
218
+ fn=check_input_token_length,
219
+ inputs=[saved_input, chatbot, system_prompt],
220
+ api_name=False,
221
+ queue=False,
222
+ ).success(
223
+ fn=generate,
224
+ inputs=[
225
+ saved_input,
226
+ chatbot,
227
+ system_prompt,
228
+ max_new_tokens,
229
+ temperature,
230
+ top_p,
231
+ top_k,
232
+ ],
233
+ outputs=chatbot,
234
+ api_name=False,
235
+ )
236
+
237
+ button_event_preprocess = (
238
+ submit_button.click(
239
+ fn=clear_and_save_textbox,
240
+ inputs=textbox,
241
+ outputs=[textbox, saved_input],
242
+ api_name=False,
243
+ queue=False,
244
+ )
245
+ .then(
246
+ fn=display_input,
247
+ inputs=[saved_input, chatbot],
248
+ outputs=chatbot,
249
+ api_name=False,
250
+ queue=False,
251
+ )
252
+ .then(
253
+ fn=check_input_token_length,
254
+ inputs=[saved_input, chatbot, system_prompt],
255
+ api_name=False,
256
+ queue=False,
257
+ )
258
+ .success(
259
+ fn=generate,
260
+ inputs=[
261
+ saved_input,
262
+ chatbot,
263
+ system_prompt,
264
+ max_new_tokens,
265
+ temperature,
266
+ top_p,
267
+ top_k,
268
+ ],
269
+ outputs=chatbot,
270
+ api_name=False,
271
+ )
272
+ )
273
+
274
+ retry_button.click(
275
+ fn=delete_prev_fn,
276
+ inputs=chatbot,
277
+ outputs=[chatbot, saved_input],
278
+ api_name=False,
279
+ queue=False,
280
+ ).then(
281
+ fn=display_input,
282
+ inputs=[saved_input, chatbot],
283
+ outputs=chatbot,
284
+ api_name=False,
285
+ queue=False,
286
+ ).then(
287
+ fn=generate,
288
+ inputs=[
289
+ saved_input,
290
+ chatbot,
291
+ system_prompt,
292
+ max_new_tokens,
293
+ temperature,
294
+ top_p,
295
+ top_k,
296
+ ],
297
+ outputs=chatbot,
298
+ api_name=False,
299
+ )
300
+
301
+ undo_button.click(
302
+ fn=delete_prev_fn,
303
+ inputs=chatbot,
304
+ outputs=[chatbot, saved_input],
305
+ api_name=False,
306
+ queue=False,
307
+ ).then(
308
+ fn=lambda x: x,
309
+ inputs=[saved_input],
310
+ outputs=textbox,
311
+ api_name=False,
312
+ queue=False,
313
+ )
314
+
315
+ clear_button.click(
316
+ fn=lambda: ([], ""),
317
+ outputs=[chatbot, saved_input],
318
+ queue=False,
319
+ api_name=False,
320
+ )
321
+
322
+ demo.queue(max_size=20).launch()
app_4bit_ggml.py ADDED
@@ -0,0 +1,320 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import argparse
2
+
3
+ import os
4
+ from typing import Iterator
5
+
6
+ import gradio as gr
7
+
8
+ # from dotenv import load_dotenv
9
+ from distutils.util import strtobool
10
+
11
+ from llama2_wrapper import LLAMA2_WRAPPER
12
+
13
+
14
+ parser = argparse.ArgumentParser()
15
+
16
+ DEFAULT_SYSTEM_PROMPT = "You are a helpful, respectful and honest assistant. Always answer as helpfully as possible, while being safe. Your answers should not include any harmful, unethical, racist, sexist, toxic, dangerous, or illegal content. Please ensure that your responses are socially unbiased and positive in nature. If a question does not make any sense, or is not factually coherent, explain why instead of answering something not correct. If you don't know the answer to a question, please don't share false information."
17
+
18
+ parser.add_argument('--model_path', type=str, required=True, default='',
19
+ help='model_path .')
20
+
21
+ parser.add_argument('--system_prompt', type=str, required=False, default=DEFAULT_SYSTEM_PROMPT,
22
+ help='Inference server Appkey. Default is .')
23
+
24
+ parser.add_argument('--max_max_new_tokens', type=int, default=2048, metavar='NUMBER',
25
+ help='maximum new tokens (default: 2048)')
26
+
27
+ FLAGS = parser.parse_args()
28
+
29
+
30
+ DEFAULT_SYSTEM_PROMPT = FLAGS.system_prompt
31
+ MAX_MAX_NEW_TOKENS = FLAGS.max_max_new_tokens
32
+
33
+ DEFAULT_MAX_NEW_TOKENS = 1024
34
+ MAX_INPUT_TOKEN_LENGTH = 4000
35
+
36
+ MODEL_PATH = FLAGS.model_path
37
+ assert MODEL_PATH is not None, f"MODEL_PATH is required, got: {MODEL_PATH}"
38
+
39
+ LOAD_IN_8BIT = False
40
+
41
+ LOAD_IN_4BIT = True
42
+
43
+ LLAMA_CPP = True
44
+
45
+ if LLAMA_CPP:
46
+ print("Running on CPU with llama.cpp.")
47
+ else:
48
+ import torch
49
+
50
+ if torch.cuda.is_available():
51
+ print("Running on GPU with torch transformers.")
52
+ else:
53
+ print("CUDA not found.")
54
+
55
+ config = {
56
+ "model_name": MODEL_PATH,
57
+ "load_in_8bit": LOAD_IN_8BIT,
58
+ "load_in_4bit": LOAD_IN_4BIT,
59
+ "llama_cpp": LLAMA_CPP,
60
+ "MAX_INPUT_TOKEN_LENGTH": MAX_INPUT_TOKEN_LENGTH,
61
+ }
62
+ llama2_wrapper = LLAMA2_WRAPPER(config)
63
+ llama2_wrapper.init_tokenizer()
64
+ llama2_wrapper.init_model()
65
+
66
+ DESCRIPTION = """
67
+ # Llama2-Chinese-7b-webui
68
+
69
+ 这是一个[Llama2-Chinese-2-7b](https://github.com/FlagAlpha/Llama2-Chinese)的推理界面。
70
+ - 支持的模型: [Llama-2-GGML](https://huggingface.co/FlagAlpha/Llama2-Chinese-7b-Chat-GGML)
71
+ - 支持的后端
72
+ - CPU(at least 6 GB RAM), Mac/AMD
73
+ """
74
+
75
+
76
+ def clear_and_save_textbox(message: str) -> tuple[str, str]:
77
+ return "", message
78
+
79
+
80
+ def display_input(
81
+ message: str, history: list[tuple[str, str]]
82
+ ) -> list[tuple[str, str]]:
83
+ history.append((message, ""))
84
+ return history
85
+
86
+
87
+ def delete_prev_fn(history: list[tuple[str, str]]) -> tuple[list[tuple[str, str]], str]:
88
+ try:
89
+ message, _ = history.pop()
90
+ except IndexError:
91
+ message = ""
92
+ return history, message or ""
93
+
94
+
95
+ def generate(
96
+ message: str,
97
+ history_with_input: list[tuple[str, str]],
98
+ system_prompt: str,
99
+ max_new_tokens: int,
100
+ temperature: float,
101
+ top_p: float,
102
+ top_k: int,
103
+ ) -> Iterator[list[tuple[str, str]]]:
104
+ if max_new_tokens > MAX_MAX_NEW_TOKENS:
105
+ raise ValueError
106
+
107
+ history = history_with_input[:-1]
108
+ generator = llama2_wrapper.run(
109
+ message, history, system_prompt, max_new_tokens, temperature, top_p, top_k
110
+ )
111
+ try:
112
+ first_response = next(generator)
113
+ yield history + [(message, first_response)]
114
+ except StopIteration:
115
+ yield history + [(message, "")]
116
+ for response in generator:
117
+ yield history + [(message, response)]
118
+
119
+
120
+ def process_example(message: str) -> tuple[str, list[tuple[str, str]]]:
121
+ generator = generate(message, [], DEFAULT_SYSTEM_PROMPT, 1024, 1, 0.95, 50)
122
+ for x in generator:
123
+ pass
124
+ return "", x
125
+
126
+
127
+ def check_input_token_length(
128
+ message: str, chat_history: list[tuple[str, str]], system_prompt: str
129
+ ) -> None:
130
+ input_token_length = llama2_wrapper.get_input_token_length(
131
+ message, chat_history, system_prompt
132
+ )
133
+ if input_token_length > MAX_INPUT_TOKEN_LENGTH:
134
+ raise gr.Error(
135
+ f"The accumulated input is too long ({input_token_length} > {MAX_INPUT_TOKEN_LENGTH}). Clear your chat history and try again."
136
+ )
137
+
138
+
139
+ with gr.Blocks(css="style.css") as demo:
140
+ gr.Markdown(DESCRIPTION)
141
+
142
+ with gr.Group():
143
+ chatbot = gr.Chatbot(label="Chatbot")
144
+ with gr.Row():
145
+ textbox = gr.Textbox(
146
+ container=False,
147
+ show_label=False,
148
+ placeholder="Type a message...",
149
+ scale=10,
150
+ )
151
+ submit_button = gr.Button("Submit", variant="primary", scale=1, min_width=0)
152
+ with gr.Row():
153
+ retry_button = gr.Button("🔄 Retry", variant="secondary")
154
+ undo_button = gr.Button("↩️ Undo", variant="secondary")
155
+ clear_button = gr.Button("🗑️ Clear", variant="secondary")
156
+
157
+ saved_input = gr.State()
158
+
159
+ with gr.Accordion(label="Advanced options", open=False):
160
+ system_prompt = gr.Textbox(
161
+ label="System prompt", value=DEFAULT_SYSTEM_PROMPT, lines=6
162
+ )
163
+ max_new_tokens = gr.Slider(
164
+ label="Max new tokens",
165
+ minimum=1,
166
+ maximum=MAX_MAX_NEW_TOKENS,
167
+ step=1,
168
+ value=DEFAULT_MAX_NEW_TOKENS,
169
+ )
170
+ temperature = gr.Slider(
171
+ label="Temperature",
172
+ minimum=0.1,
173
+ maximum=4.0,
174
+ step=0.1,
175
+ value=1.0,
176
+ )
177
+ top_p = gr.Slider(
178
+ label="Top-p (nucleus sampling)",
179
+ minimum=0.05,
180
+ maximum=1.0,
181
+ step=0.05,
182
+ value=0.95,
183
+ )
184
+ top_k = gr.Slider(
185
+ label="Top-k",
186
+ minimum=1,
187
+ maximum=1000,
188
+ step=1,
189
+ value=50,
190
+ )
191
+
192
+ gr.Examples(
193
+ examples=[
194
+ "Hello there! How are you doing?",
195
+ "Can you explain briefly to me what is the Python programming language?",
196
+ ],
197
+ inputs=textbox,
198
+ outputs=[textbox, chatbot],
199
+ fn=process_example,
200
+ cache_examples=True,
201
+ )
202
+
203
+ textbox.submit(
204
+ fn=clear_and_save_textbox,
205
+ inputs=textbox,
206
+ outputs=[textbox, saved_input],
207
+ api_name=False,
208
+ queue=False,
209
+ ).then(
210
+ fn=display_input,
211
+ inputs=[saved_input, chatbot],
212
+ outputs=chatbot,
213
+ api_name=False,
214
+ queue=False,
215
+ ).then(
216
+ fn=check_input_token_length,
217
+ inputs=[saved_input, chatbot, system_prompt],
218
+ api_name=False,
219
+ queue=False,
220
+ ).success(
221
+ fn=generate,
222
+ inputs=[
223
+ saved_input,
224
+ chatbot,
225
+ system_prompt,
226
+ max_new_tokens,
227
+ temperature,
228
+ top_p,
229
+ top_k,
230
+ ],
231
+ outputs=chatbot,
232
+ api_name=False,
233
+ )
234
+
235
+ button_event_preprocess = (
236
+ submit_button.click(
237
+ fn=clear_and_save_textbox,
238
+ inputs=textbox,
239
+ outputs=[textbox, saved_input],
240
+ api_name=False,
241
+ queue=False,
242
+ )
243
+ .then(
244
+ fn=display_input,
245
+ inputs=[saved_input, chatbot],
246
+ outputs=chatbot,
247
+ api_name=False,
248
+ queue=False,
249
+ )
250
+ .then(
251
+ fn=check_input_token_length,
252
+ inputs=[saved_input, chatbot, system_prompt],
253
+ api_name=False,
254
+ queue=False,
255
+ )
256
+ .success(
257
+ fn=generate,
258
+ inputs=[
259
+ saved_input,
260
+ chatbot,
261
+ system_prompt,
262
+ max_new_tokens,
263
+ temperature,
264
+ top_p,
265
+ top_k,
266
+ ],
267
+ outputs=chatbot,
268
+ api_name=False,
269
+ )
270
+ )
271
+
272
+ retry_button.click(
273
+ fn=delete_prev_fn,
274
+ inputs=chatbot,
275
+ outputs=[chatbot, saved_input],
276
+ api_name=False,
277
+ queue=False,
278
+ ).then(
279
+ fn=display_input,
280
+ inputs=[saved_input, chatbot],
281
+ outputs=chatbot,
282
+ api_name=False,
283
+ queue=False,
284
+ ).then(
285
+ fn=generate,
286
+ inputs=[
287
+ saved_input,
288
+ chatbot,
289
+ system_prompt,
290
+ max_new_tokens,
291
+ temperature,
292
+ top_p,
293
+ top_k,
294
+ ],
295
+ outputs=chatbot,
296
+ api_name=False,
297
+ )
298
+
299
+ undo_button.click(
300
+ fn=delete_prev_fn,
301
+ inputs=chatbot,
302
+ outputs=[chatbot, saved_input],
303
+ api_name=False,
304
+ queue=False,
305
+ ).then(
306
+ fn=lambda x: x,
307
+ inputs=[saved_input],
308
+ outputs=textbox,
309
+ api_name=False,
310
+ queue=False,
311
+ )
312
+
313
+ clear_button.click(
314
+ fn=lambda: ([], ""),
315
+ outputs=[chatbot, saved_input],
316
+ queue=False,
317
+ api_name=False,
318
+ )
319
+
320
+ demo.queue(max_size=20).launch(server_name="0.0.0.0", server_port=8090, share=True)
benchmark.py ADDED
@@ -0,0 +1,96 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import os
2
+ import time
3
+
4
+ from dotenv import load_dotenv
5
+ from distutils.util import strtobool
6
+
7
+ from llama2_wrapper import LLAMA2_WRAPPER
8
+
9
+
10
+ def main():
11
+ load_dotenv()
12
+
13
+ DEFAULT_SYSTEM_PROMPT = (
14
+ os.getenv("DEFAULT_SYSTEM_PROMPT")
15
+ if os.getenv("DEFAULT_SYSTEM_PROMPT") is not None
16
+ else ""
17
+ )
18
+ MAX_MAX_NEW_TOKENS = (
19
+ int(os.getenv("MAX_MAX_NEW_TOKENS"))
20
+ if os.getenv("DEFAULT_MAX_NEW_TOKENS") is not None
21
+ else 2048
22
+ )
23
+ DEFAULT_MAX_NEW_TOKENS = (
24
+ int(os.getenv("DEFAULT_MAX_NEW_TOKENS"))
25
+ if os.getenv("DEFAULT_MAX_NEW_TOKENS") is not None
26
+ else 1024
27
+ )
28
+ MAX_INPUT_TOKEN_LENGTH = (
29
+ int(os.getenv("MAX_INPUT_TOKEN_LENGTH"))
30
+ if os.getenv("MAX_INPUT_TOKEN_LENGTH") is not None
31
+ else 4000
32
+ )
33
+
34
+ MODEL_PATH = os.getenv("MODEL_PATH")
35
+ assert MODEL_PATH is not None, f"MODEL_PATH is required, got: {MODEL_PATH}"
36
+
37
+ LOAD_IN_8BIT = bool(strtobool(os.getenv("LOAD_IN_8BIT", "True")))
38
+
39
+ LOAD_IN_4BIT = bool(strtobool(os.getenv("LOAD_IN_4BIT", "True")))
40
+
41
+ LLAMA_CPP = bool(strtobool(os.getenv("LLAMA_CPP", "True")))
42
+
43
+ if LLAMA_CPP:
44
+ print("Running on CPU with llama.cpp.")
45
+ else:
46
+ import torch
47
+
48
+ if torch.cuda.is_available():
49
+ print("Running on GPU with torch transformers.")
50
+ else:
51
+ print("CUDA not found.")
52
+
53
+ config = {
54
+ "model_name": MODEL_PATH,
55
+ "load_in_8bit": LOAD_IN_8BIT,
56
+ "load_in_4bit": LOAD_IN_4BIT,
57
+ "llama_cpp": LLAMA_CPP,
58
+ "MAX_INPUT_TOKEN_LENGTH": MAX_INPUT_TOKEN_LENGTH,
59
+ }
60
+
61
+ tic = time.perf_counter()
62
+ llama2_wrapper = LLAMA2_WRAPPER(config)
63
+ llama2_wrapper.init_tokenizer()
64
+ llama2_wrapper.init_model()
65
+ toc = time.perf_counter()
66
+ print(f"Initialize the model in {toc - tic:0.4f} seconds.")
67
+
68
+ example = "Can you explain briefly to me what is the Python programming language?"
69
+
70
+ generator = llama2_wrapper.run(
71
+ example, [], DEFAULT_SYSTEM_PROMPT, DEFAULT_MAX_NEW_TOKENS, 1, 0.95, 50
72
+ )
73
+ tic = time.perf_counter()
74
+ try:
75
+ first_response = next(generator)
76
+ # history += [(example, first_response)]
77
+ # print(first_response)
78
+ except StopIteration:
79
+ pass
80
+ # history += [(example, "")]
81
+ # print(history)
82
+ for response in generator:
83
+ # history += [(example, response)]
84
+ # print(response)
85
+ pass
86
+ print(response)
87
+
88
+ toc = time.perf_counter()
89
+ output_token_length = llama2_wrapper.get_token_length(response)
90
+
91
+ print(f"Generating the out in {toc - tic:0.4f} seconds.")
92
+ print(f"Speed: {output_token_length / (toc - tic):0.4f} tokens/sec.")
93
+
94
+
95
+ if __name__ == "__main__":
96
+ main()
docs/performance.md ADDED
@@ -0,0 +1,19 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # Benchmark Performance
2
+
3
+ ## Performance on Nvidia GPU
4
+
5
+ | Model | Precision | Device | GPU VRAM | Speed (tokens / sec) | load time (s) |
6
+ | --------------------------------- | --------- | ---------- | ---------------------- | ---------------- | ---------------- |
7
+ | Llama-2-7b-chat-hf | 16 bit | | | | |
8
+ | Llama-2-7b-chat-hf | 8bit | NVIDIA RTX 2080 Ti | 7.7 GB VRAM | 3.76 | 783.87 |
9
+ | Llama-2-7b-Chat-GPTQ | 4 bit | NVIDIA RTX 2080 Ti | 5.8 GB VRAM | 12.08 | 192.91 |
10
+ | Llama-2-13b-chat-hf | 16 bit | | | | |
11
+ | | | | | | |
12
+
13
+ ## Performance on CPU / OpenBLAS / cuBLAS / CLBlast / Metal
14
+
15
+ | Model | Precision | Device | RAM / GPU VRAM | Speed (tokens / sec) | load time (s) |
16
+ | --------------------------------- | --------- | ---------- | ---------------------- | ---------------- | ---------------- |
17
+ | Llama-2-7B-Chat-GGML | 4 bit | Intel i7-8700 | 5.1GB RAM | 4.16 | 105.75 |
18
+ | Llama-2-7B-Chat-GGML | 4 bit | Apple M1 CPU | | | |
19
+
env_examples/.env.13b_example ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ MODEL_PATH = "/path-to/Llama-2-13b-chat-hf"
2
+ LOAD_IN_8BIT = False
3
+ LOAD_IN_4BIT = False
4
+ LLAMA_CPP = False
5
+
6
+ MAX_MAX_NEW_TOKENS = 2048
7
+ DEFAULT_MAX_NEW_TOKENS = 1024
8
+ MAX_INPUT_TOKEN_LENGTH = 4000
9
+
10
+ DEFAULT_SYSTEM_PROMPT = "You are a helpful, respectful and honest assistant. Always answer as helpfully as possible, while being safe. Your answers should not include any harmful, unethical, racist, sexist, toxic, dangerous, or illegal content. Please ensure that your responses are socially unbiased and positive in nature. If a question does not make any sense, or is not factually coherent, explain why instead of answering something not correct. If you don't know the answer to a question, please don't share false information."
env_examples/.env.7b_8bit_example ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ MODEL_PATH = "/path-to/Llama-2-7b-chat-hf"
2
+ LOAD_IN_8BIT = True
3
+ LOAD_IN_4BIT = False
4
+ LLAMA_CPP = False
5
+
6
+ MAX_MAX_NEW_TOKENS = 2048
7
+ DEFAULT_MAX_NEW_TOKENS = 1024
8
+ MAX_INPUT_TOKEN_LENGTH = 4000
9
+
10
+ DEFAULT_SYSTEM_PROMPT = "You are a helpful, respectful and honest assistant. Always answer as helpfully as possible, while being safe. Your answers should not include any harmful, unethical, racist, sexist, toxic, dangerous, or illegal content. Please ensure that your responses are socially unbiased and positive in nature. If a question does not make any sense, or is not factually coherent, explain why instead of answering something not correct. If you don't know the answer to a question, please don't share false information."
env_examples/.env.7b_ggmlv3_q4_0_example ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ MODEL_PATH = "/path-to/llama-2-7b-chat.ggmlv3.q4_0.bin"
2
+ LOAD_IN_8BIT = False
3
+ LOAD_IN_4BIT = True
4
+ LLAMA_CPP = True
5
+
6
+ MAX_MAX_NEW_TOKENS = 2048
7
+ DEFAULT_MAX_NEW_TOKENS = 1024
8
+ MAX_INPUT_TOKEN_LENGTH = 4000
9
+
10
+ DEFAULT_SYSTEM_PROMPT = "You are a helpful, respectful and honest assistant. Always answer as helpfully as possible, while being safe. Your answers should not include any harmful, unethical, racist, sexist, toxic, dangerous, or illegal content. Please ensure that your responses are socially unbiased and positive in nature. If a question does not make any sense, or is not factually coherent, explain why instead of answering something not correct. If you don't know the answer to a question, please don't share false information."
env_examples/.env.7b_gptq_example ADDED
@@ -0,0 +1,10 @@
 
 
 
 
 
 
 
 
 
 
 
1
+ MODEL_PATH = "/path-to/Llama-2-7b-Chat-GPTQ"
2
+ LOAD_IN_8BIT = False
3
+ LOAD_IN_4BIT = True
4
+ LLAMA_CPP = False
5
+
6
+ MAX_MAX_NEW_TOKENS = 2048
7
+ DEFAULT_MAX_NEW_TOKENS = 1024
8
+ MAX_INPUT_TOKEN_LENGTH = 4000
9
+
10
+ DEFAULT_SYSTEM_PROMPT = "You are a helpful, respectful and honest assistant. Always answer as helpfully as possible, while being safe. Your answers should not include any harmful, unethical, racist, sexist, toxic, dangerous, or illegal content. Please ensure that your responses are socially unbiased and positive in nature. If a question does not make any sense, or is not factually coherent, explain why instead of answering something not correct. If you don't know the answer to a question, please don't share false information."
gradio_cached_examples/19/Chatbot/tmpihfsul2n.json ADDED
@@ -0,0 +1 @@
 
 
1
+ [["Hello there! How are you doing?", "I am an artificial intelligence language model assistant and do not have emotions or feelings as humans do. I am here to help answer any questions in a safe, socially unbiased and positive nature, respectfully. What is on your mind that I can assist you with today?</s></s>"]]
gradio_cached_examples/19/Chatbot/tmpj22ucqjj.json ADDED
@@ -0,0 +1 @@
 
 
1
+ [["Can you explain briefly to me what is the Python programming language?", "The Python programming language is a general-purpose, high-level, and versatile programming language that was developed in the late 1980s by Guido van Rossum. It's widely used for various tasks, including scientific computing, web development, data analysis, artificial intelligence, and more.\n\nPython has built-in support for object-oriented programming, which makes it easy to write elegant and modular code. Its syntax is simple and intuitive, making it an excellent choice for beginners as well as experienced programmers. Furthermore, Python has an extensive standard library that provides functionality for various tasks like reading and writing files, handling exceptions, working with regular expressions, connecting to databases, and more. \n\nOverall, the Python language is widely recognized as a \"megabyte of cheese\" - it's easy on your stomach!</s>"]]
gradio_cached_examples/19/log.csv ADDED
@@ -0,0 +1,3 @@
 
 
 
 
1
+ component 0,Chatbot,flag,username,timestamp
2
+ ,/root/llama2-webui/gradio_cached_examples/19/Chatbot/tmpihfsul2n.json,,,2023-07-29 15:34:58.289613
3
+ ,/root/llama2-webui/gradio_cached_examples/19/Chatbot/tmpj22ucqjj.json,,,2023-07-29 15:38:58.697165
llama2_wrapper/__init__.py ADDED
@@ -0,0 +1 @@
 
 
1
+ from .model import LLAMA2_WRAPPER, get_prompt
llama2_wrapper/__pycache__/__init__.cpython-310.pyc ADDED
Binary file (203 Bytes). View file
 
llama2_wrapper/__pycache__/model.cpython-310.pyc ADDED
Binary file (5.18 kB). View file
 
llama2_wrapper/model.py ADDED
@@ -0,0 +1,197 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ # coding:utf-8
2
+ from threading import Thread
3
+ from typing import Any, Iterator
4
+
5
+
6
+ class LLAMA2_WRAPPER:
7
+ def __init__(self, config: dict = {}):
8
+ self.config = config
9
+ self.model = None
10
+ self.tokenizer = None
11
+
12
+ def init_model(self):
13
+ if self.model is None:
14
+ self.model = LLAMA2_WRAPPER.create_llama2_model(
15
+ self.config,
16
+ )
17
+ if not self.config.get("llama_cpp"):
18
+ self.model.eval()
19
+
20
+ def init_tokenizer(self):
21
+ if self.tokenizer is None and not self.config.get("llama_cpp"):
22
+ self.tokenizer = LLAMA2_WRAPPER.create_llama2_tokenizer(self.config)
23
+
24
+ @classmethod
25
+ def create_llama2_model(cls, config):
26
+ model_name = config.get("model_name")
27
+ load_in_8bit = config.get("load_in_8bit", True)
28
+ load_in_4bit = config.get("load_in_4bit", False)
29
+ llama_cpp = config.get("llama_cpp", False)
30
+ if llama_cpp:
31
+ from llama_cpp import Llama
32
+
33
+ model = Llama(
34
+ model_path=model_name,
35
+ n_ctx=config.get("MAX_INPUT_TOKEN_LENGTH"),
36
+ n_batch=config.get("MAX_INPUT_TOKEN_LENGTH"),
37
+ )
38
+ elif load_in_4bit:
39
+ from auto_gptq import AutoGPTQForCausalLM
40
+
41
+ model = AutoGPTQForCausalLM.from_quantized(
42
+ model_name,
43
+ use_safetensors=True,
44
+ trust_remote_code=True,
45
+ device="cuda:0",
46
+ use_triton=False,
47
+ quantize_config=None,
48
+ )
49
+ else:
50
+ import torch
51
+ from transformers import AutoModelForCausalLM
52
+
53
+ model = AutoModelForCausalLM.from_pretrained(
54
+ model_name,
55
+ device_map="auto",
56
+ torch_dtype=torch.float16,
57
+ load_in_8bit=load_in_8bit,
58
+ )
59
+ return model
60
+
61
+ @classmethod
62
+ def create_llama2_tokenizer(cls, config):
63
+ model_name = config.get("model_name")
64
+ from transformers import AutoTokenizer
65
+
66
+ tokenizer = AutoTokenizer.from_pretrained(model_name)
67
+ return tokenizer
68
+
69
+ def get_token_length(
70
+ self,
71
+ prompt: str,
72
+ ) -> int:
73
+ if self.config.get("llama_cpp"):
74
+ input_ids = self.model.tokenize(bytes(prompt, "utf-8"))
75
+ return len(input_ids)
76
+ else:
77
+ input_ids = self.tokenizer([prompt], return_tensors="np")["input_ids"]
78
+ return input_ids.shape[-1]
79
+
80
+ def get_input_token_length(
81
+ self, message: str, chat_history: list[tuple[str, str]], system_prompt: str
82
+ ) -> int:
83
+ prompt = get_prompt(message, chat_history, system_prompt)
84
+
85
+ return self.get_token_length(prompt)
86
+
87
+ def generate(
88
+ self,
89
+ prompt: str,
90
+ max_new_tokens: int = 1024,
91
+ temperature: float = 0.8,
92
+ top_p: float = 0.95,
93
+ top_k: int = 50,
94
+ ) -> Iterator[str]:
95
+ if self.config.get("llama_cpp"):
96
+ inputs = self.model.tokenize(bytes(prompt, "utf-8"))
97
+ generate_kwargs = dict(
98
+ top_p=top_p,
99
+ top_k=top_k,
100
+ temp=temperature,
101
+ )
102
+
103
+ generator = self.model.generate(inputs, **generate_kwargs)
104
+ outputs = []
105
+ answer_message =''
106
+ new_tokens = []
107
+ for token in generator:
108
+ if token!='</s>':
109
+ try:
110
+ new_tokens.append(token)
111
+ b_text = self.model.detokenize(new_tokens)
112
+ # b_text = self.model.decode(new_tokens)
113
+ answer_message+=str(b_text, encoding="utf-8")
114
+ new_tokens = []
115
+ except:
116
+ pass
117
+ else:
118
+ yield answer_message
119
+ break
120
+
121
+ if 'Human:' in answer_message:
122
+ answer_message = answer_message.split('Human:')[0]
123
+ yield answer_message
124
+ break
125
+
126
+ if token == self.model.token_eos():
127
+ yield answer_message
128
+ break
129
+
130
+ yield answer_message
131
+ else:
132
+ from transformers import TextIteratorStreamer
133
+
134
+ inputs = self.tokenizer([prompt], return_tensors="pt").to("cuda")
135
+
136
+ streamer = TextIteratorStreamer(
137
+ self.tokenizer, timeout=10.0, skip_prompt=True, skip_special_tokens=True
138
+ )
139
+ generate_kwargs = dict(
140
+ inputs,
141
+ streamer=streamer,
142
+ max_new_tokens=max_new_tokens,
143
+ do_sample=True,
144
+ top_p=top_p,
145
+ top_k=top_k,
146
+ temperature=temperature,
147
+ num_beams=1,
148
+ )
149
+ t = Thread(target=self.model.generate, kwargs=generate_kwargs)
150
+ t.start()
151
+
152
+ outputs = []
153
+ for text in streamer:
154
+ outputs.append(text)
155
+ yield "".join(outputs)
156
+
157
+ def run(
158
+ self,
159
+ message: str,
160
+ chat_history: list[tuple[str, str]],
161
+ system_prompt: str,
162
+ max_new_tokens: int = 1024,
163
+ temperature: float = 0.3,
164
+ top_p: float = 0.95,
165
+ top_k: int = 50,
166
+ ) -> Iterator[str]:
167
+ prompt = get_prompt(message, chat_history, system_prompt)
168
+ return self.generate(prompt, max_new_tokens, temperature, top_p, top_k)
169
+
170
+ def __call__(
171
+ self,
172
+ prompt: str,
173
+ **kwargs: Any,
174
+ ) -> str:
175
+ if self.config.get("llama_cpp"):
176
+ return self.model.__call__(prompt, **kwargs)["choices"][0]["text"]
177
+ else:
178
+ inputs = self.tokenizer([prompt], return_tensors="pt").input_ids.to("cuda")
179
+ output = self.model.generate(inputs=inputs, **kwargs)
180
+ return self.tokenizer.decode(output[0])
181
+
182
+
183
+ def get_prompt(
184
+ message: str, chat_history: list[tuple[str, str]], system_prompt: str
185
+ ) -> str:
186
+ prompt = ''
187
+ for user_input, response in chat_history:
188
+ prompt += "<s>Human: " + user_input.strip()+"\n</s><s>Assistant: " + response.strip()+"\n</s>"
189
+
190
+ prompt += "<s>Human: " + message.strip() +"\n</s><s>Assistant: "
191
+ prompt = prompt[-2048:]
192
+
193
+ if len(system_prompt)>0:
194
+ prompt = '<s>System: '+system_prompt.strip()+'\n</s>'+ prompt
195
+ return prompt
196
+
197
+
poetry.lock ADDED
The diff for this file is too large to render. See raw diff
 
pyproject.toml ADDED
@@ -0,0 +1,33 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ [tool.poetry]
2
+ name = "llama2-wrapper"
3
+ version = "0.1.3"
4
+ description = "Running Llama 2 on GPU or CPU from anywhere (Linux/Windows/Mac)."
5
+ authors = ["liltom-eth <liltom.eth@gmail.com>"]
6
+ license = "MIT"
7
+ homepage = "https://github.com/liltom-eth/llama2-webui"
8
+ repository = "https://github.com/liltom-eth/llama2-webui"
9
+
10
+ packages = [{include = "llama2_wrapper"}]
11
+
12
+ [tool.poetry.dependencies]
13
+ python = ">=3.10,<3.13"
14
+ accelerate = "^0.21.0"
15
+ auto-gptq = "0.3.0"
16
+ gradio = "3.37.0"
17
+ protobuf = "3.20.3"
18
+ scipy = "1.11.1"
19
+ sentencepiece = "0.1.99"
20
+ torch = "2.0.1"
21
+ transformers = "4.31.0"
22
+ tqdm = "4.65.0"
23
+ python-dotenv = "1.0.0"
24
+ llama-cpp-python = "^0.1.77"
25
+ bitsandbytes = [
26
+ {platform = 'linux', version = "0.40.2"},
27
+ {platform = 'darwin', version = "0.40.2"},
28
+ ]
29
+
30
+
31
+ [build-system]
32
+ requires = ["poetry-core"]
33
+ build-backend = "poetry.core.masonry.api"
requirements.txt ADDED
@@ -0,0 +1,12 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ accelerate==0.21.0
2
+ auto-gptq==0.3.0
3
+ bitsandbytes==0.40.2
4
+ gradio==3.37.0
5
+ protobuf==3.20.3
6
+ scipy==1.11.1
7
+ sentencepiece==0.1.99
8
+ torch==2.0.1
9
+ transformers==4.31.0
10
+ tqdm==4.65.0
11
+ python-dotenv==1.0.0
12
+ llama-cpp-python== 0.1.77
static/screenshot.png ADDED
tests/__init__.py ADDED
File without changes