From 8b45547c09fe63ad0cb61dc13a5e26927fba943c Mon Sep 17 00:00:00 2001 From: Wonderplex <50866817+Jasonqi146@users.noreply.github.com> Date: Mon, 6 Nov 2023 20:23:55 -0500 Subject: [PATCH] added llama-factory under llm_rl (#87) --- .gitignore | 162 +++- llm_rl/LICENSE | 201 +++++ llm_rl/README.md | 501 ++++++++++++ llm_rl/README_zh.md | 500 ++++++++++++ llm_rl/assets/wechat.jpg | Bin 0 -> 143444 bytes llm_rl/pyproject.toml | 3 + llm_rl/requirements.txt | 20 + llm_rl/reward_model.sh | 21 + llm_rl/setup.py | 55 ++ llm_rl/src/api_demo.py | 14 + llm_rl/src/cli_demo.py | 39 + llm_rl/src/evaluate.py | 190 +++++ llm_rl/src/export_model.py | 9 + llm_rl/src/llmtuner/__init__.py | 9 + llm_rl/src/llmtuner/api/__init__.py | 1 + llm_rl/src/llmtuner/api/app.py | 146 ++++ llm_rl/src/llmtuner/api/protocol.py | 83 ++ llm_rl/src/llmtuner/chat/__init__.py | 1 + llm_rl/src/llmtuner/chat/stream_chat.py | 109 +++ llm_rl/src/llmtuner/dsets/__init__.py | 3 + llm_rl/src/llmtuner/dsets/loader.py | 145 ++++ llm_rl/src/llmtuner/dsets/preprocess.py | 268 +++++++ llm_rl/src/llmtuner/dsets/utils.py | 59 ++ llm_rl/src/llmtuner/extras/__init__.py | 0 llm_rl/src/llmtuner/extras/callbacks.py | 155 ++++ llm_rl/src/llmtuner/extras/constants.py | 92 +++ llm_rl/src/llmtuner/extras/logging.py | 43 ++ llm_rl/src/llmtuner/extras/misc.py | 118 +++ .../src/llmtuner/extras/patches/__init__.py | 0 .../llmtuner/extras/patches/llama_patch.py | 218 ++++++ llm_rl/src/llmtuner/extras/ploting.py | 52 ++ llm_rl/src/llmtuner/extras/save_and_load.py | 21 + llm_rl/src/llmtuner/extras/template.py | 713 ++++++++++++++++++ llm_rl/src/llmtuner/hparams/__init__.py | 4 + llm_rl/src/llmtuner/hparams/data_args.py | 169 +++++ .../src/llmtuner/hparams/finetuning_args.py | 107 +++ llm_rl/src/llmtuner/hparams/general_args.py | 13 + .../src/llmtuner/hparams/generating_args.py | 53 ++ llm_rl/src/llmtuner/hparams/model_args.py | 93 +++ llm_rl/src/llmtuner/tuner/__init__.py | 1 + llm_rl/src/llmtuner/tuner/core/__init__.py | 2 + llm_rl/src/llmtuner/tuner/core/adapter.py | 101 +++ llm_rl/src/llmtuner/tuner/core/loader.py | 244 ++++++ llm_rl/src/llmtuner/tuner/core/parser.py | 226 ++++++ llm_rl/src/llmtuner/tuner/core/utils.py | 94 +++ llm_rl/src/llmtuner/tuner/dpo/__init__.py | 1 + llm_rl/src/llmtuner/tuner/dpo/collator.py | 51 ++ llm_rl/src/llmtuner/tuner/dpo/trainer.py | 104 +++ llm_rl/src/llmtuner/tuner/dpo/workflow.py | 66 ++ llm_rl/src/llmtuner/tuner/ppo/__init__.py | 1 + llm_rl/src/llmtuner/tuner/ppo/trainer.py | 310 ++++++++ llm_rl/src/llmtuner/tuner/ppo/utils.py | 35 + llm_rl/src/llmtuner/tuner/ppo/workflow.py | 92 +++ llm_rl/src/llmtuner/tuner/pt/__init__.py | 1 + llm_rl/src/llmtuner/tuner/pt/workflow.py | 58 ++ llm_rl/src/llmtuner/tuner/rm/__init__.py | 1 + llm_rl/src/llmtuner/tuner/rm/collator.py | 27 + llm_rl/src/llmtuner/tuner/rm/metric.py | 7 + llm_rl/src/llmtuner/tuner/rm/trainer.py | 105 +++ llm_rl/src/llmtuner/tuner/rm/workflow.py | 68 ++ llm_rl/src/llmtuner/tuner/sft/__init__.py | 1 + llm_rl/src/llmtuner/tuner/sft/metric.py | 53 ++ llm_rl/src/llmtuner/tuner/sft/trainer.py | 92 +++ llm_rl/src/llmtuner/tuner/sft/workflow.py | 90 +++ llm_rl/src/llmtuner/tuner/tune.py | 51 ++ llm_rl/src/llmtuner/webui/__init__.py | 1 + llm_rl/src/llmtuner/webui/chatter.py | 101 +++ llm_rl/src/llmtuner/webui/common.py | 103 +++ .../src/llmtuner/webui/components/__init__.py | 6 + .../src/llmtuner/webui/components/chatbot.py | 49 ++ llm_rl/src/llmtuner/webui/components/data.py | 103 +++ llm_rl/src/llmtuner/webui/components/eval.py | 70 ++ .../src/llmtuner/webui/components/export.py | 79 ++ llm_rl/src/llmtuner/webui/components/infer.py | 39 + llm_rl/src/llmtuner/webui/components/top.py | 74 ++ llm_rl/src/llmtuner/webui/components/train.py | 154 ++++ llm_rl/src/llmtuner/webui/css.py | 20 + llm_rl/src/llmtuner/webui/engine.py | 57 ++ llm_rl/src/llmtuner/webui/interface.py | 66 ++ llm_rl/src/llmtuner/webui/locales.py | 698 +++++++++++++++++ llm_rl/src/llmtuner/webui/manager.py | 35 + llm_rl/src/llmtuner/webui/runner.py | 254 +++++++ llm_rl/src/llmtuner/webui/utils.py | 85 +++ llm_rl/src/train_bash.py | 14 + llm_rl/src/train_web.py | 11 + llm_rl/src/web_demo.py | 11 + llm_rl/tests/cal_flops.py | 44 ++ llm_rl/tests/llamafy_baichuan2.py | 86 +++ llm_rl/tests/llamafy_qwen.py | 135 ++++ llm_rl/tests/quantize.py | 50 ++ 90 files changed, 8616 insertions(+), 1 deletion(-) create mode 100644 llm_rl/LICENSE create mode 100644 llm_rl/README.md create mode 100644 llm_rl/README_zh.md create mode 100644 llm_rl/assets/wechat.jpg create mode 100644 llm_rl/pyproject.toml create mode 100644 llm_rl/requirements.txt create mode 100644 llm_rl/reward_model.sh create mode 100644 llm_rl/setup.py create mode 100644 llm_rl/src/api_demo.py create mode 100644 llm_rl/src/cli_demo.py create mode 100644 llm_rl/src/evaluate.py create mode 100644 llm_rl/src/export_model.py create mode 100644 llm_rl/src/llmtuner/__init__.py create mode 100644 llm_rl/src/llmtuner/api/__init__.py create mode 100644 llm_rl/src/llmtuner/api/app.py create mode 100644 llm_rl/src/llmtuner/api/protocol.py create mode 100644 llm_rl/src/llmtuner/chat/__init__.py create mode 100644 llm_rl/src/llmtuner/chat/stream_chat.py create mode 100644 llm_rl/src/llmtuner/dsets/__init__.py create mode 100644 llm_rl/src/llmtuner/dsets/loader.py create mode 100644 llm_rl/src/llmtuner/dsets/preprocess.py create mode 100644 llm_rl/src/llmtuner/dsets/utils.py create mode 100644 llm_rl/src/llmtuner/extras/__init__.py create mode 100644 llm_rl/src/llmtuner/extras/callbacks.py create mode 100644 llm_rl/src/llmtuner/extras/constants.py create mode 100644 llm_rl/src/llmtuner/extras/logging.py create mode 100644 llm_rl/src/llmtuner/extras/misc.py create mode 100644 llm_rl/src/llmtuner/extras/patches/__init__.py create mode 100644 llm_rl/src/llmtuner/extras/patches/llama_patch.py create mode 100644 llm_rl/src/llmtuner/extras/ploting.py create mode 100644 llm_rl/src/llmtuner/extras/save_and_load.py create mode 100644 llm_rl/src/llmtuner/extras/template.py create mode 100644 llm_rl/src/llmtuner/hparams/__init__.py create mode 100644 llm_rl/src/llmtuner/hparams/data_args.py create mode 100644 llm_rl/src/llmtuner/hparams/finetuning_args.py create mode 100644 llm_rl/src/llmtuner/hparams/general_args.py create mode 100644 llm_rl/src/llmtuner/hparams/generating_args.py create mode 100644 llm_rl/src/llmtuner/hparams/model_args.py create mode 100644 llm_rl/src/llmtuner/tuner/__init__.py create mode 100644 llm_rl/src/llmtuner/tuner/core/__init__.py create mode 100644 llm_rl/src/llmtuner/tuner/core/adapter.py create mode 100644 llm_rl/src/llmtuner/tuner/core/loader.py create mode 100644 llm_rl/src/llmtuner/tuner/core/parser.py create mode 100644 llm_rl/src/llmtuner/tuner/core/utils.py create mode 100644 llm_rl/src/llmtuner/tuner/dpo/__init__.py create mode 100644 llm_rl/src/llmtuner/tuner/dpo/collator.py create mode 100644 llm_rl/src/llmtuner/tuner/dpo/trainer.py create mode 100644 llm_rl/src/llmtuner/tuner/dpo/workflow.py create mode 100644 llm_rl/src/llmtuner/tuner/ppo/__init__.py create mode 100644 llm_rl/src/llmtuner/tuner/ppo/trainer.py create mode 100644 llm_rl/src/llmtuner/tuner/ppo/utils.py create mode 100644 llm_rl/src/llmtuner/tuner/ppo/workflow.py create mode 100644 llm_rl/src/llmtuner/tuner/pt/__init__.py create mode 100644 llm_rl/src/llmtuner/tuner/pt/workflow.py create mode 100644 llm_rl/src/llmtuner/tuner/rm/__init__.py create mode 100644 llm_rl/src/llmtuner/tuner/rm/collator.py create mode 100644 llm_rl/src/llmtuner/tuner/rm/metric.py create mode 100644 llm_rl/src/llmtuner/tuner/rm/trainer.py create mode 100644 llm_rl/src/llmtuner/tuner/rm/workflow.py create mode 100644 llm_rl/src/llmtuner/tuner/sft/__init__.py create mode 100644 llm_rl/src/llmtuner/tuner/sft/metric.py create mode 100644 llm_rl/src/llmtuner/tuner/sft/trainer.py create mode 100644 llm_rl/src/llmtuner/tuner/sft/workflow.py create mode 100644 llm_rl/src/llmtuner/tuner/tune.py create mode 100644 llm_rl/src/llmtuner/webui/__init__.py create mode 100644 llm_rl/src/llmtuner/webui/chatter.py create mode 100644 llm_rl/src/llmtuner/webui/common.py create mode 100644 llm_rl/src/llmtuner/webui/components/__init__.py create mode 100644 llm_rl/src/llmtuner/webui/components/chatbot.py create mode 100644 llm_rl/src/llmtuner/webui/components/data.py create mode 100644 llm_rl/src/llmtuner/webui/components/eval.py create mode 100644 llm_rl/src/llmtuner/webui/components/export.py create mode 100644 llm_rl/src/llmtuner/webui/components/infer.py create mode 100644 llm_rl/src/llmtuner/webui/components/top.py create mode 100644 llm_rl/src/llmtuner/webui/components/train.py create mode 100644 llm_rl/src/llmtuner/webui/css.py create mode 100644 llm_rl/src/llmtuner/webui/engine.py create mode 100644 llm_rl/src/llmtuner/webui/interface.py create mode 100644 llm_rl/src/llmtuner/webui/locales.py create mode 100644 llm_rl/src/llmtuner/webui/manager.py create mode 100644 llm_rl/src/llmtuner/webui/runner.py create mode 100644 llm_rl/src/llmtuner/webui/utils.py create mode 100644 llm_rl/src/train_bash.py create mode 100644 llm_rl/src/train_web.py create mode 100644 llm_rl/src/web_demo.py create mode 100644 llm_rl/tests/cal_flops.py create mode 100644 llm_rl/tests/llamafy_baichuan2.py create mode 100644 llm_rl/tests/llamafy_qwen.py create mode 100644 llm_rl/tests/quantize.py diff --git a/.gitignore b/.gitignore index 3e563d1d..83339037 100644 --- a/.gitignore +++ b/.gitignore @@ -5,6 +5,13 @@ __pycache__ dist .venv +# Byte-compiled / optimized / DLL files +*.py[cod] +*$py.class + +# C extensions +*.so + # Log *.log *.log.* @@ -33,4 +40,157 @@ tests/state_of_the_union.txt # Build build -!dummy_file \ No newline at end of file +!dummy_file + +# Distribution / packaging +.Python +build/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib/ +lib64/ +parts/ +sdist/ +var/ +wheels/ +share/python-wheels/ +*.egg-info/ +.installed.cfg +*.egg +MANIFEST + +# PyInstaller +# Usually these files are written by a python script from a template +# before PyInstaller builds the exe, so as to inject date/other infos into it. +*.manifest +*.spec + +# Installer logs +pip-log.txt +pip-delete-this-directory.txt + +# Unit test / coverage reports +htmlcov/ +.tox/ +.nox/ +.coverage +.coverage.* +.cache +nosetests.xml +coverage.xml +*.cover +*.py,cover +.hypothesis/ +.pytest_cache/ +cover/ + +# Translations +*.mo +*.pot + +# Django stuff: +*.log +local_settings.py +db.sqlite3 +db.sqlite3-journal + +# Flask stuff: +instance/ +.webassets-cache + +# Scrapy stuff: +.scrapy + +# Sphinx documentation +docs/_build/ + +# PyBuilder +.pybuilder/ +target/ + +# Jupyter Notebook +.ipynb_checkpoints + +# IPython +profile_default/ +ipython_config.py + +# pyenv +# For a library or package, you might want to ignore these files since the code is +# intended to run in multiple environments; otherwise, check them in: +# .python-version + +# pipenv +# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. +# However, in case of collaboration, if having platform-specific dependencies or dependencies +# having no cross-platform support, pipenv may install dependencies that don't work, or not +# install all needed dependencies. +#Pipfile.lock + +# poetry +# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control. +# This is especially recommended for binary packages to ensure reproducibility, and is more +# commonly ignored for libraries. +# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control +#poetry.lock + +# pdm +# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control. +#pdm.lock +# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it +# in version control. +# https://pdm.fming.dev/#use-with-ide +.pdm.toml + +# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm +__pypackages__/ + +# Celery stuff +celerybeat-schedule +celerybeat.pid + +# SageMath parsed files +*.sage.py + +# Environments +.env +.venv +env/ +venv/ +ENV/ +env.bak/ +venv.bak/ + +# Spyder project settings +.spyderproject +.spyproject + +# Rope project settings +.ropeproject + +# mkdocs documentation +/site + +# mypy +.mypy_cache/ +.dmypy.json +dmypy.json + +# Pyre type checker +.pyre/ + +# pytype static type analyzer +.pytype/ + +# Cython debug symbols +cython_debug/ + +# PyCharm +# JetBrains specific template is maintained in a separate JetBrains.gitignore that can +# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore +# and can be added to the global gitignore or merged into this file. For a more nuclear +# option (not recommended) you can uncomment the following to ignore the entire idea folder. +#.idea/ \ No newline at end of file diff --git a/llm_rl/LICENSE b/llm_rl/LICENSE new file mode 100644 index 00000000..b09cd785 --- /dev/null +++ b/llm_rl/LICENSE @@ -0,0 +1,201 @@ +Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + 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. diff --git a/llm_rl/README.md b/llm_rl/README.md new file mode 100644 index 00000000..54da92da --- /dev/null +++ b/llm_rl/README.md @@ -0,0 +1,501 @@ +# LLaMA Factory: Training and Evaluating Large Language Models with Minimal Effort + +[![GitHub Repo stars](https://img.shields.io/github/stars/hiyouga/LLaMA-Factory?style=social)](https://github.com/hiyouga/LLaMA-Factory/stargazers) +[![GitHub Code License](https://img.shields.io/github/license/hiyouga/LLaMA-Factory)](LICENSE) +[![GitHub last commit](https://img.shields.io/github/last-commit/hiyouga/LLaMA-Factory)](https://github.com/hiyouga/LLaMA-Factory/commits/main) +[![PyPI](https://img.shields.io/pypi/v/llmtuner)](https://pypi.org/project/llmtuner/) +[![Downloads](https://static.pepy.tech/badge/llmtuner)](https://pypi.org/project/llmtuner/) +[![GitHub pull request](https://img.shields.io/badge/PRs-welcome-blue)](https://github.com/hiyouga/LLaMA-Factory/pulls) +[![Discord](https://dcbadge.vercel.app/api/server/e73gccsSd?compact=true&style=flat)](https://discord.gg/e73gccsSd) + +👋 Join our [WeChat](assets/wechat.jpg). + +\[ English | [中文](README_zh.md) \] + +## LLaMA Board: A One-stop Web UI for Getting Started with LLaMA Factory + +Launch **LLaMA Board** via `CUDA_VISIBLE_DEVICES=0 python src/train_web.py`. (multiple GPUs are not supported yet) + +Here is an example of altering the self-cognition of an instruction-tuned language model within 10 minutes on a single GPU. + +https://github.com/hiyouga/LLaMA-Factory/assets/16256802/6ba60acc-e2e2-4bec-b846-2d88920d5ba1 + +## Changelog + +[23/10/21] We supported **[NEFTune](https://arxiv.org/abs/2310.05914)** trick for fine-tuning. Try `--neft_alpha` argument to activate NEFTune, e.g., `--neft_alpha 5`. + +[23/09/27] We supported **$S^2$-Attn** proposed by [LongLoRA](https://github.com/dvlab-research/LongLoRA) for the LLaMA models. Try `--shift_attn` argument to enable shift short attention. + +[23/09/23] We integrated MMLU, C-Eval and CMMLU benchmarks in this repo. See [this example](#evaluation) to evaluate your models. + +[23/09/10] We supported using **[FlashAttention-2](https://github.com/Dao-AILab/flash-attention)** for the LLaMA models. Try `--flash_attn` argument to enable FlashAttention-2 if you are using RTX4090, A100 or H100 GPUs. + +[23/08/12] We supported **RoPE scaling** to extend the context length of the LLaMA models. Try `--rope_scaling linear` argument in training and `--rope_scaling dynamic` argument at inference to extrapolate the position embeddings. + +[23/08/11] We supported **[DPO training](https://arxiv.org/abs/2305.18290)** for instruction-tuned models. See [this example](#dpo-training) to train your models. + +[23/07/31] We supported **dataset streaming**. Try `--streaming` and `--max_steps 10000` arguments to load your dataset in streaming mode. + +[23/07/29] We released two instruction-tuned 13B models at Hugging Face. See these Hugging Face Repos ([LLaMA-2](https://huggingface.co/hiyouga/Llama-2-Chinese-13b-chat) / [Baichuan](https://huggingface.co/hiyouga/Baichuan-13B-sft)) for details. + +[23/07/18] We developed an **all-in-one Web UI** for training, evaluation and inference. Try `train_web.py` to fine-tune models in your Web browser. Thank [@KanadeSiina](https://github.com/KanadeSiina) and [@codemayq](https://github.com/codemayq) for their efforts in the development. + +[23/07/09] We released **[FastEdit](https://github.com/hiyouga/FastEdit)** ⚡🩹, an easy-to-use package for editing the factual knowledge of large language models efficiently. Please follow [FastEdit](https://github.com/hiyouga/FastEdit) if you are interested. + +[23/06/29] We provided a **reproducible example** of training a chat model using instruction-following datasets, see [Baichuan-7B-sft](https://huggingface.co/hiyouga/Baichuan-7B-sft) for details. + +[23/06/22] We aligned the [demo API](src/api_demo.py) with the [OpenAI's](https://platform.openai.com/docs/api-reference/chat) format where you can insert the fine-tuned model in **arbitrary ChatGPT-based applications**. + +[23/06/03] We supported quantized training and inference (aka **[QLoRA](https://github.com/artidoro/qlora)**). Try `--quantization_bit 4/8` argument to work with quantized models. + +## Supported Models + +| Model | Model size | Default module | Template | +| -------------------------------------------------------- | --------------------------- | ----------------- | --------- | +| [Baichuan](https://github.com/baichuan-inc/Baichuan-13B) | 7B/13B | W_pack | baichuan | +| [Baichuan2](https://github.com/baichuan-inc/Baichuan2) | 7B/13B | W_pack | baichuan2 | +| [BLOOM](https://huggingface.co/bigscience/bloom) | 560M/1.1B/1.7B/3B/7.1B/176B | query_key_value | - | +| [BLOOMZ](https://huggingface.co/bigscience/bloomz) | 560M/1.1B/1.7B/3B/7.1B/176B | query_key_value | - | +| [ChatGLM3](https://github.com/THUDM/ChatGLM3) | 6B | query_key_value | chatglm3 | +| [Falcon](https://huggingface.co/tiiuae/falcon-7b) | 7B/40B/180B | query_key_value | - | +| [InternLM](https://github.com/InternLM/InternLM) | 7B/20B | q_proj,v_proj | intern | +| [LLaMA](https://github.com/facebookresearch/llama) | 7B/13B/33B/65B | q_proj,v_proj | - | +| [LLaMA-2](https://huggingface.co/meta-llama) | 7B/13B/70B | q_proj,v_proj | llama2 | +| [Mistral](https://huggingface.co/mistralai) | 7B | q_proj,v_proj | mistral | +| [Phi-1.5](https://huggingface.co/microsoft/phi-1_5) | 1.3B | Wqkv | - | +| [Qwen](https://github.com/QwenLM/Qwen-7B) | 7B/14B | c_attn | qwen | +| [XVERSE](https://github.com/xverse-ai) | 7B/13B/65B | q_proj,v_proj | xverse | + +> [!NOTE] +> **Default module** is used for the `--lora_target` argument, you can use `--lora_target all` to specify all the available modules. +> +> For the "base" models, the `--template` argument can be chosen from `default`, `alpaca`, `vicuna` etc. But make sure to use the **corresponding template** for the "chat" models. + +Please refer to [template.py](src/llmtuner/extras/template.py) for a full list of models we supported. + +## Supported Training Approaches + +| Approach | Full-parameter | Partial-parameter | LoRA | QLoRA | +| ---------------------- | ------------------ | ------------------ | ------------------ | ------------------ | +| Pre-Training | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | +| Supervised Fine-Tuning | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | +| Reward Modeling | | | :white_check_mark: | :white_check_mark: | +| PPO Training | | | :white_check_mark: | :white_check_mark: | +| DPO Training | :white_check_mark: | | :white_check_mark: | :white_check_mark: | + +> [!NOTE] +> Use `--quantization_bit 4/8` argument to enable QLoRA. + +## Provided Datasets + +
Pre-training datasets + +- [Wiki Demo (en)](data/wiki_demo.txt) +- [RefinedWeb (en)](https://huggingface.co/datasets/tiiuae/falcon-refinedweb) +- [RedPajama V2 (en)](https://huggingface.co/datasets/togethercomputer/RedPajama-Data-V2) +- [Wikipedia (en)](https://huggingface.co/datasets/olm/olm-wikipedia-20221220) +- [Wikipedia (zh)](https://huggingface.co/datasets/pleisto/wikipedia-cn-20230720-filtered) +- [Pile (en)](https://huggingface.co/datasets/EleutherAI/pile) +- [SkyPile (zh)](https://huggingface.co/datasets/Skywork/SkyPile-150B) +- [The Stack (en)](https://huggingface.co/datasets/bigcode/the-stack) +- [StarCoder (en)](https://huggingface.co/datasets/bigcode/starcoderdata) + +
+ +
Supervised fine-tuning datasets + +- [Stanford Alpaca (en)](https://github.com/tatsu-lab/stanford_alpaca) +- [Stanford Alpaca (zh)](https://github.com/ymcui/Chinese-LLaMA-Alpaca) +- [GPT-4 Generated Data (en&zh)](https://github.com/Instruction-Tuning-with-GPT-4/GPT-4-LLM) +- [Self-cognition (zh)](data/self_cognition.json) +- [Open Assistant (multilingual)](https://huggingface.co/datasets/OpenAssistant/oasst1) +- [ShareGPT (zh)](https://huggingface.co/datasets/QingyiSi/Alpaca-CoT/tree/main/Chinese-instruction-collection) +- [Guanaco Dataset (multilingual)](https://huggingface.co/datasets/JosephusCheung/GuanacoDataset) +- [BELLE 2M (zh)](https://huggingface.co/datasets/BelleGroup/train_2M_CN) +- [BELLE 1M (zh)](https://huggingface.co/datasets/BelleGroup/train_1M_CN) +- [BELLE 0.5M (zh)](https://huggingface.co/datasets/BelleGroup/train_0.5M_CN) +- [BELLE Dialogue 0.4M (zh)](https://huggingface.co/datasets/BelleGroup/generated_chat_0.4M) +- [BELLE School Math 0.25M (zh)](https://huggingface.co/datasets/BelleGroup/school_math_0.25M) +- [BELLE Multiturn Chat 0.8M (zh)](https://huggingface.co/datasets/BelleGroup/multiturn_chat_0.8M) +- [UltraChat (en)](https://github.com/thunlp/UltraChat) +- [LIMA (en)](https://huggingface.co/datasets/GAIR/lima) +- [OpenPlatypus (en)](https://huggingface.co/datasets/garage-bAInd/Open-Platypus) +- [CodeAlpaca 20k (en)](https://huggingface.co/datasets/sahil2801/CodeAlpaca-20k) +- [Alpaca CoT (multilingual)](https://huggingface.co/datasets/QingyiSi/Alpaca-CoT) +- [MathInstruct (en)](https://huggingface.co/datasets/TIGER-Lab/MathInstruct) +- [Firefly 1.1M (zh)](https://huggingface.co/datasets/YeungNLP/firefly-train-1.1M) +- [Web QA (zh)](https://huggingface.co/datasets/suolyer/webqa) +- [WebNovel (zh)](https://huggingface.co/datasets/zxbsmk/webnovel_cn) +- [Ad Gen (zh)](https://huggingface.co/datasets/HasturOfficial/adgen) +- [ShareGPT Hyperfiltered (en)](https://huggingface.co/datasets/totally-not-an-llm/sharegpt-hyperfiltered-3k) +- [ShareGPT4 (en&zh)](https://huggingface.co/datasets/shibing624/sharegpt_gpt4) +- [UltraChat 200k (en)](https://huggingface.co/datasets/HuggingFaceH4/ultrachat_200k) +- [AgentInstruct (en)](https://huggingface.co/datasets/THUDM/AgentInstruct) +- [LMSYS Chat 1M (en)](https://huggingface.co/datasets/lmsys/lmsys-chat-1m) +- [Evol Instruct V2 (en)](https://huggingface.co/datasets/WizardLM/WizardLM_evol_instruct_V2_196k) + +
+ +
Preference datasets + +- [HH-RLHF (en)](https://huggingface.co/datasets/Anthropic/hh-rlhf) +- [Open Assistant (multilingual)](https://huggingface.co/datasets/OpenAssistant/oasst1) +- [GPT-4 Generated Data (en&zh)](https://github.com/Instruction-Tuning-with-GPT-4/GPT-4-LLM) + +
+ +Please refer to [data/README.md](data/README.md) for details. + +Some datasets require confirmation before using them, so we recommend logging in with your Hugging Face account using these commands. + +```bash +pip install --upgrade huggingface_hub +huggingface-cli login +``` + +## Requirement + +- Python 3.8+ and PyTorch 1.13.1+ +- 🤗Transformers, Datasets, Accelerate, PEFT and TRL +- sentencepiece, protobuf and tiktoken +- fire, jieba, rouge-chinese and nltk (used at evaluation and predict) +- gradio and matplotlib (used in web UI) +- uvicorn, fastapi and sse-starlette (used in API) + +And **powerful GPUs**! + +## Getting Started + +### Data Preparation (optional) + +Please refer to [data/README.md](data/README.md) for checking the details about the format of dataset files. You can either use a single `.json` file or a [dataset loading script](https://huggingface.co/docs/datasets/dataset_script) with multiple files to create a custom dataset. + +> [!NOTE] +> Please update `data/dataset_info.json` to use your custom dataset. About the format of this file, please refer to `data/README.md`. + +### Dependence Installation (optional) + +```bash +git clone https://github.com/hiyouga/LLaMA-Factory.git +conda create -n llama_factory python=3.10 +conda activate llama_factory +cd LLaMA-Factory +pip install -r requirements.txt +``` + +If you want to enable the quantized LoRA (QLoRA) on the Windows platform, you will be required to install a pre-built version of `bitsandbytes` library, which supports CUDA 11.1 to 12.1. + +```bash +pip install https://github.com/jllllll/bitsandbytes-windows-webui/releases/download/wheels/bitsandbytes-0.39.1-py3-none-win_amd64.whl +``` + +### Train on a single GPU + +> [!IMPORTANT] +> If you want to train models on multiple GPUs, please refer to [Distributed Training](#distributed-training). + +#### Pre-Training + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage pt \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset wiki_demo \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --output_dir path_to_pt_checkpoint \ + --overwrite_cache \ + --per_device_train_batch_size 4 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 5e-5 \ + --num_train_epochs 3.0 \ + --plot_loss \ + --fp16 +``` + +#### Supervised Fine-Tuning + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage sft \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset alpaca_gpt4_en \ + --template default \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --output_dir path_to_sft_checkpoint \ + --overwrite_cache \ + --per_device_train_batch_size 4 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 5e-5 \ + --num_train_epochs 3.0 \ + --plot_loss \ + --fp16 +``` + +#### Reward Modeling + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage rm \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset comparison_gpt4_en \ + --template default \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --resume_lora_training False \ + --checkpoint_dir path_to_sft_checkpoint \ + --output_dir path_to_rm_checkpoint \ + --per_device_train_batch_size 2 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 1e-6 \ + --num_train_epochs 1.0 \ + --plot_loss \ + --fp16 +``` + +#### PPO Training + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage ppo \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset alpaca_gpt4_en \ + --template default \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --resume_lora_training False \ + --checkpoint_dir path_to_sft_checkpoint \ + --reward_model path_to_rm_checkpoint \ + --output_dir path_to_ppo_checkpoint \ + --per_device_train_batch_size 2 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 1e-5 \ + --num_train_epochs 1.0 \ + --plot_loss \ + --fp16 +``` + +#### DPO Training + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage dpo \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset comparison_gpt4_en \ + --template default \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --resume_lora_training False \ + --checkpoint_dir path_to_sft_checkpoint \ + --output_dir path_to_dpo_checkpoint \ + --per_device_train_batch_size 2 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 1e-5 \ + --num_train_epochs 1.0 \ + --plot_loss \ + --fp16 +``` + +### Distributed Training + +#### Use Huggingface Accelerate + +```bash +accelerate config # configure the environment +accelerate launch src/train_bash.py # arguments (same as above) +``` + +
Example config for LoRA training + +```yaml +compute_environment: LOCAL_MACHINE +distributed_type: MULTI_GPU +downcast_bf16: 'no' +gpu_ids: all +machine_rank: 0 +main_training_function: main +mixed_precision: fp16 +num_machines: 1 +num_processes: 4 +rdzv_backend: static +same_network: true +tpu_env: [] +tpu_use_cluster: false +tpu_use_sudo: false +use_cpu: false +``` + +
+ +#### Use DeepSpeed + +```bash +deepspeed --num_gpus 8 --master_port=9901 src/train_bash.py \ + --deepspeed ds_config.json \ + ... # arguments (same as above) +``` + +
Example config for full-parameter training with DeepSpeed ZeRO-2 + +```json +{ + "train_batch_size": "auto", + "train_micro_batch_size_per_gpu": "auto", + "gradient_accumulation_steps": "auto", + "gradient_clipping": "auto", + "zero_allow_untested_optimizer": true, + "fp16": { + "enabled": "auto", + "loss_scale": 0, + "initial_scale_power": 16, + "loss_scale_window": 1000, + "hysteresis": 2, + "min_loss_scale": 1 + }, + "zero_optimization": { + "stage": 2, + "allgather_partitions": true, + "allgather_bucket_size": 5e8, + "reduce_scatter": true, + "reduce_bucket_size": 5e8, + "overlap_comm": false, + "contiguous_gradients": true + } +} +``` + +
+ +### Export model + +```bash +python src/export_model.py \ + --model_name_or_path path_to_llama_model \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint \ + --export_dir path_to_export +``` + +### API Demo + +```bash +python src/api_demo.py \ + --model_name_or_path path_to_llama_model \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint +``` + +> [!NOTE] +> Visit `http://localhost:8000/docs` for API documentation. + +### CLI Demo + +```bash +python src/cli_demo.py \ + --model_name_or_path path_to_llama_model \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint +``` + +### Web Demo + +```bash +python src/web_demo.py \ + --model_name_or_path path_to_llama_model \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint +``` + +### Evaluation + +```bash +CUDA_VISIBLE_DEVICES=0 python src/evaluate.py \ + --model_name_or_path path_to_llama_model \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint \ + --template vanilla \ + --task mmlu \ + --split test \ + --lang en \ + --n_shot 5 \ + --batch_size 4 +``` + +### Predict + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage sft \ + --model_name_or_path path_to_llama_model \ + --do_predict \ + --dataset alpaca_gpt4_en \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint \ + --output_dir path_to_predict_result \ + --per_device_eval_batch_size 8 \ + --max_samples 100 \ + --predict_with_generate +``` + +> [!NOTE] +> We recommend using `--per_device_eval_batch_size=1` and `--max_target_length 128` at 4/8-bit predict. + +## Projects using LLaMA Factory + +- **[StarWhisper](https://github.com/Yu-Yang-Li/StarWhisper)**: A large language model for Astronomy, based on ChatGLM2-6B and Qwen-14B. +- **[DISC-LawLLM](https://github.com/FudanDISC/DISC-LawLLM)**: A large language model specialized in Chinese legal domain, based on Baichuan-13B, is capable of retrieving and reasoning on legal knowledge. +- **[Sunsimiao](https://github.com/thomas-yanxin/Sunsimiao)**: A large language model specialized in Chinese medical domain, based on Baichuan-7B and ChatGLM-6B. +- **[CareGPT](https://github.com/WangRongsheng/CareGPT)**: A series of large language models for Chinese medical domain, based on LLaMA2-7B and Baichuan-13B. + +## License + +This repository is licensed under the [Apache-2.0 License](LICENSE). + +Please follow the model licenses to use the corresponding model weights: [Baichuan](https://huggingface.co/baichuan-inc/Baichuan-13B-Base/resolve/main/Community%20License%20for%20Baichuan-13B%20Model.pdf) / [Baichuan2](https://huggingface.co/baichuan-inc/Baichuan2-13B-Chat/resolve/main/Community%20License%20for%20Baichuan2%20Model.pdf) / [BLOOM](https://huggingface.co/spaces/bigscience/license) / [ChatGLM3](https://github.com/THUDM/ChatGLM3/blob/main/MODEL_LICENSE) / [Falcon](https://huggingface.co/tiiuae/falcon-180B/blob/main/LICENSE.txt) / [InternLM](https://github.com/InternLM/InternLM#license) / [LLaMA](https://github.com/facebookresearch/llama/blob/main/MODEL_CARD.md) / [LLaMA-2](https://ai.meta.com/llama/license/) / [Mistral](LICENSE) / [Phi-1.5](https://huggingface.co/microsoft/phi-1_5/resolve/main/Research%20License.docx) / [Qwen](https://huggingface.co/Qwen/Qwen-7B-Chat/blob/main/LICENSE) / [XVERSE](https://github.com/xverse-ai/XVERSE-13B/blob/main/MODEL_LICENSE.pdf) + +## Citation + +If this work is helpful, please kindly cite as: + +```bibtex +@Misc{llama-factory, + title = {LLaMA Factory}, + author = {hiyouga}, + howpublished = {\url{https://github.com/hiyouga/LLaMA-Factory}}, + year = {2023} +} +``` + +## Acknowledgement + +This repo benefits from [PEFT](https://github.com/huggingface/peft), [QLoRA](https://github.com/artidoro/qlora) and [FastChat](https://github.com/lm-sys/FastChat). Thanks for their wonderful works. + +## Star History + +![Star History Chart](https://api.star-history.com/svg?repos=hiyouga/LLaMA-Factory&type=Date) diff --git a/llm_rl/README_zh.md b/llm_rl/README_zh.md new file mode 100644 index 00000000..c69e3983 --- /dev/null +++ b/llm_rl/README_zh.md @@ -0,0 +1,500 @@ +# LLaMA Factory: 轻松的大模型训练与评估 + +[![GitHub Repo stars](https://img.shields.io/github/stars/hiyouga/LLaMA-Factory?style=social)](https://github.com/hiyouga/LLaMA-Factory/stargazers) +[![GitHub Code License](https://img.shields.io/github/license/hiyouga/LLaMA-Factory)](LICENSE) +[![GitHub last commit](https://img.shields.io/github/last-commit/hiyouga/LLaMA-Factory)](https://github.com/hiyouga/LLaMA-Factory/commits/main) +[![PyPI](https://img.shields.io/pypi/v/llmtuner)](https://pypi.org/project/llmtuner/) +[![Downloads](https://static.pepy.tech/badge/llmtuner)](https://pypi.org/project/llmtuner/) +[![GitHub pull request](https://img.shields.io/badge/PRs-welcome-blue)](https://github.com/hiyouga/LLaMA-Factory/pulls) +[![Discord](https://dcbadge.vercel.app/api/server/e73gccsSd?compact=true&style=flat)](https://discord.gg/e73gccsSd) + +👋 加入我们的[微信群](assets/wechat.jpg)。 + +\[ [English](README.md) | 中文 \] + +## LLaMA Board: 通过一站式网页界面快速上手 LLaMA Factory + +使用 `CUDA_VISIBLE_DEVICES=0 python src/train_web.py` 启动 **LLaMA Board**。(该界面目前仅支持单卡训练) + +下面是使用单张 GPU 在 10 分钟内更改对话式大型语言模型自我认知的示例。 + +https://github.com/hiyouga/LLaMA-Factory/assets/16256802/6ba60acc-e2e2-4bec-b846-2d88920d5ba1 + +## 更新日志 + +[23/10/21] 我们支持了 **[NEFTune](https://arxiv.org/abs/2310.05914)** 训练技巧。请使用 `--neft_alpha` 参数启用 NEFTune,例如 `--neft_alpha 5`。 + +[23/09/27] 我们针对 LLaMA 模型支持了 [LongLoRA](https://github.com/dvlab-research/LongLoRA) 提出的 **$S^2$-Attn**。请使用 `--shift_attn` 参数以启用该功能。 + +[23/09/23] 我们在项目中集成了 MMLU、C-Eval 和 CMMLU 评估集。使用方法请参阅[此示例](#模型评估)。 + +[23/09/10] 我们针对 LLaMA 模型支持了 **[FlashAttention-2](https://github.com/Dao-AILab/flash-attention)**。如果您使用的是 RTX4090、A100 或 H100 GPU,请使用 `--flash_attn` 参数以启用 FlashAttention-2。 + +[23/08/12] 我们支持了 **RoPE 插值**来扩展 LLaMA 模型的上下文长度。请使用 `--rope_scaling linear` 参数训练模型或使用 `--rope_scaling dynamic` 参数评估模型。 + +[23/08/11] 我们支持了指令模型的 **[DPO 训练](https://arxiv.org/abs/2305.18290)**。使用方法请参阅[此示例](#dpo-训练)。 + +[23/07/31] 我们支持了**数据流式加载**。请尝试使用 `--streaming` 和 `--max_steps 10000` 参数来流式加载数据集。 + +[23/07/29] 我们在 Hugging Face 发布了两个 13B 指令微调模型。详细内容请查阅我们的 Hugging Face 项目([LLaMA-2](https://huggingface.co/hiyouga/Llama-2-Chinese-13b-chat) / [Baichuan](https://huggingface.co/hiyouga/Baichuan-13B-sft))。 + +[23/07/18] 我们开发了支持训练和测试的**浏览器一体化界面**。请使用 `train_web.py` 在您的浏览器中微调模型。感谢 [@KanadeSiina](https://github.com/KanadeSiina) 和 [@codemayq](https://github.com/codemayq) 在该功能开发中付出的努力。 + +[23/07/09] 我们开源了 **[FastEdit](https://github.com/hiyouga/FastEdit)** ⚡🩹,一个简单易用的、能迅速编辑大模型事实记忆的工具包。如果您感兴趣请关注我们的 [FastEdit](https://github.com/hiyouga/FastEdit) 项目。 + +[23/06/29] 我们提供了一个**可复现的**指令模型微调示例,详细内容请查阅 [Baichuan-7B-sft](https://huggingface.co/hiyouga/Baichuan-7B-sft)。 + +[23/06/22] 我们对齐了[示例 API](src/api_demo.py) 与 [OpenAI API](https://platform.openai.com/docs/api-reference/chat) 的格式,您可以将微调模型接入**任意基于 ChatGPT 的应用**中。 + +[23/06/03] 我们实现了 4 比特的 LoRA 训练(也称 **[QLoRA](https://github.com/artidoro/qlora)**)。请使用 `--quantization_bit 4` 参数进行 4 比特量化微调。 + +## 模型 + +| 模型名 | 模型大小 | 默认模块 | Template | +| -------------------------------------------------------- | --------------------------- | ----------------- | --------- | +| [Baichuan](https://github.com/baichuan-inc/Baichuan-13B) | 7B/13B | W_pack | baichuan | +| [Baichuan2](https://github.com/baichuan-inc/Baichuan2) | 7B/13B | W_pack | baichuan2 | +| [BLOOM](https://huggingface.co/bigscience/bloom) | 560M/1.1B/1.7B/3B/7.1B/176B | query_key_value | - | +| [BLOOMZ](https://huggingface.co/bigscience/bloomz) | 560M/1.1B/1.7B/3B/7.1B/176B | query_key_value | - | +| [ChatGLM3](https://github.com/THUDM/ChatGLM3) | 6B | query_key_value | chatglm3 | +| [Falcon](https://huggingface.co/tiiuae/falcon-7b) | 7B/40B/180B | query_key_value | - | +| [InternLM](https://github.com/InternLM/InternLM) | 7B/20B | q_proj,v_proj | intern | +| [LLaMA](https://github.com/facebookresearch/llama) | 7B/13B/33B/65B | q_proj,v_proj | - | +| [LLaMA-2](https://huggingface.co/meta-llama) | 7B/13B/70B | q_proj,v_proj | llama2 | +| [Mistral](https://huggingface.co/mistralai) | 7B | q_proj,v_proj | mistral | +| [Phi-1.5](https://huggingface.co/microsoft/phi-1_5) | 1.3B | Wqkv | - | +| [Qwen](https://github.com/QwenLM/Qwen-7B) | 7B/14B | c_attn | qwen | +| [XVERSE](https://github.com/xverse-ai) | 7B/13B/65B | q_proj,v_proj | xverse | + +> [!NOTE] +> **默认模块**应作为 `--lora_target` 参数的默认值,可使用 `--lora_target all` 参数指定全部模块。 +> +> 对于所有“基座”(Base)模型,`--template` 参数可以是 `default`, `alpaca`, `vicuna` 等任意值。但“对话”(Chat)模型请务必使用**对应的模板**。 + +项目所支持模型的完整列表请参阅 [template.py](src/llmtuner/extras/template.py)。 + +## 训练方法 + +| 方法 | 全参数训练 | 部分参数训练 | LoRA | QLoRA | +| ---------------------- | ------------------ | ------------------ | ------------------ | ------------------ | +| 预训练 | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | +| 指令监督微调 | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | +| 奖励模型训练 | | | :white_check_mark: | :white_check_mark: | +| PPO 训练 | | | :white_check_mark: | :white_check_mark: | +| DPO 训练 | :white_check_mark: | | :white_check_mark: | :white_check_mark: | + +> [!NOTE] +> 请使用 `--quantization_bit 4/8` 参数来启用 QLoRA 训练。 + +## 数据集 + +
预训练数据集 + +- [Wiki Demo (en)](data/wiki_demo.txt) +- [RefinedWeb (en)](https://huggingface.co/datasets/tiiuae/falcon-refinedweb) +- [RedPajama V2 (en)](https://huggingface.co/datasets/togethercomputer/RedPajama-Data-V2) +- [Wikipedia (en)](https://huggingface.co/datasets/olm/olm-wikipedia-20221220) +- [Wikipedia (zh)](https://huggingface.co/datasets/pleisto/wikipedia-cn-20230720-filtered) +- [Pile (en)](https://huggingface.co/datasets/EleutherAI/pile) +- [SkyPile (zh)](https://huggingface.co/datasets/Skywork/SkyPile-150B) +- [The Stack (en)](https://huggingface.co/datasets/bigcode/the-stack) +- [StarCoder (en)](https://huggingface.co/datasets/bigcode/starcoderdata) + +
+ +
指令微调数据集 + +- [Stanford Alpaca (en)](https://github.com/tatsu-lab/stanford_alpaca) +- [Stanford Alpaca (zh)](https://github.com/ymcui/Chinese-LLaMA-Alpaca) +- [GPT-4 Generated Data (en&zh)](https://github.com/Instruction-Tuning-with-GPT-4/GPT-4-LLM) +- [Self-cognition (zh)](data/self_cognition.json) +- [Open Assistant (multilingual)](https://huggingface.co/datasets/OpenAssistant/oasst1) +- [ShareGPT (zh)](https://huggingface.co/datasets/QingyiSi/Alpaca-CoT/tree/main/Chinese-instruction-collection) +- [Guanaco Dataset (multilingual)](https://huggingface.co/datasets/JosephusCheung/GuanacoDataset) +- [BELLE 2M (zh)](https://huggingface.co/datasets/BelleGroup/train_2M_CN) +- [BELLE 1M (zh)](https://huggingface.co/datasets/BelleGroup/train_1M_CN) +- [BELLE 0.5M (zh)](https://huggingface.co/datasets/BelleGroup/train_0.5M_CN) +- [BELLE Dialogue 0.4M (zh)](https://huggingface.co/datasets/BelleGroup/generated_chat_0.4M) +- [BELLE School Math 0.25M (zh)](https://huggingface.co/datasets/BelleGroup/school_math_0.25M) +- [BELLE Multiturn Chat 0.8M (zh)](https://huggingface.co/datasets/BelleGroup/multiturn_chat_0.8M) +- [UltraChat (en)](https://github.com/thunlp/UltraChat) +- [LIMA (en)](https://huggingface.co/datasets/GAIR/lima) +- [OpenPlatypus (en)](https://huggingface.co/datasets/garage-bAInd/Open-Platypus) +- [CodeAlpaca 20k (en)](https://huggingface.co/datasets/sahil2801/CodeAlpaca-20k) +- [Alpaca CoT (multilingual)](https://huggingface.co/datasets/QingyiSi/Alpaca-CoT) +- [MathInstruct (en)](https://huggingface.co/datasets/TIGER-Lab/MathInstruct) +- [Firefly 1.1M (zh)](https://huggingface.co/datasets/YeungNLP/firefly-train-1.1M) +- [Web QA (zh)](https://huggingface.co/datasets/suolyer/webqa) +- [WebNovel (zh)](https://huggingface.co/datasets/zxbsmk/webnovel_cn) +- [Ad Gen (zh)](https://huggingface.co/datasets/HasturOfficial/adgen) +- [ShareGPT Hyperfiltered (en)](https://huggingface.co/datasets/totally-not-an-llm/sharegpt-hyperfiltered-3k) +- [ShareGPT4 (en&zh)](https://huggingface.co/datasets/shibing624/sharegpt_gpt4) +- [UltraChat 200k (en)](https://huggingface.co/datasets/HuggingFaceH4/ultrachat_200k) +- [AgentInstruct (en)](https://huggingface.co/datasets/THUDM/AgentInstruct) +- [LMSYS Chat 1M (en)](https://huggingface.co/datasets/lmsys/lmsys-chat-1m) +- [Evol Instruct V2 (en)](https://huggingface.co/datasets/WizardLM/WizardLM_evol_instruct_V2_196k) + +
+ +
偏好数据集 + +- [HH-RLHF (en)](https://huggingface.co/datasets/Anthropic/hh-rlhf) +- [Open Assistant (multilingual)](https://huggingface.co/datasets/OpenAssistant/oasst1) +- [GPT-4 Generated Data (en&zh)](https://github.com/Instruction-Tuning-with-GPT-4/GPT-4-LLM) + +
+ +使用方法请参考 [data/README_zh.md](data/README_zh.md) 文件。 + +部分数据集的使用需要确认,我们推荐使用下述命令登录您的 Hugging Face 账户。 + +```bash +pip install --upgrade huggingface_hub +huggingface-cli login +``` + +## 软件依赖 + +- Python 3.8+ 和 PyTorch 1.13.1+ +- 🤗Transformers, Datasets, Accelerate, PEFT 和 TRL +- sentencepiece, protobuf 和 tiktoken +- fire, jieba, rouge-chinese 和 nltk (用于评估及预测) +- gradio 和 matplotlib (用于网页端交互) +- uvicorn, fastapi 和 sse-starlette (用于 API) + +以及 **强而有力的 GPU**! + +## 如何使用 + +### 数据准备(可跳过) + +关于数据集文件的格式,请参考 [data/README_zh.md](data/README_zh.md) 的内容。构建自定义数据集时,既可以使用单个 `.json` 文件,也可以使用一个[数据加载脚本](https://huggingface.co/docs/datasets/dataset_script)和多个文件。 + +> [!NOTE] +> 使用自定义数据集时,请更新 `data/dataset_info.json` 文件,该文件的格式请参考 `data/README_zh.md`。 + +### 环境搭建(可跳过) + +```bash +git clone https://github.com/hiyouga/LLaMA-Factory.git +conda create -n llama_factory python=3.10 +conda activate llama_factory +cd LLaMA-Factory +pip install -r requirements.txt +``` + +如果要在 Windows 平台上开启量化 LoRA(QLoRA),需要安装预编译的 `bitsandbytes` 库, 支持 CUDA 11.1 到 12.1. + +```bash +pip install https://github.com/jllllll/bitsandbytes-windows-webui/releases/download/wheels/bitsandbytes-0.39.1-py3-none-win_amd64.whl +``` + +### 单 GPU 训练 + +> [!IMPORTANT] +> 如果您使用多张 GPU 训练模型,请移步[多 GPU 分布式训练](#多-gpu-分布式训练)部分。 + +#### 预训练 + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage pt \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset wiki_demo \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --output_dir path_to_pt_checkpoint \ + --overwrite_cache \ + --per_device_train_batch_size 4 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 5e-5 \ + --num_train_epochs 3.0 \ + --plot_loss \ + --fp16 +``` + +#### 指令监督微调 + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage sft \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset alpaca_gpt4_zh \ + --template default \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --output_dir path_to_sft_checkpoint \ + --overwrite_cache \ + --per_device_train_batch_size 4 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 5e-5 \ + --num_train_epochs 3.0 \ + --plot_loss \ + --fp16 +``` + +#### 奖励模型训练 + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage rm \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset comparison_gpt4_zh \ + --template default \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --resume_lora_training False \ + --checkpoint_dir path_to_sft_checkpoint \ + --output_dir path_to_rm_checkpoint \ + --per_device_train_batch_size 2 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 1e-6 \ + --num_train_epochs 1.0 \ + --plot_loss \ + --fp16 +``` + +#### PPO 训练 + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage ppo \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset alpaca_gpt4_zh \ + --template default \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --resume_lora_training False \ + --checkpoint_dir path_to_sft_checkpoint \ + --reward_model path_to_rm_checkpoint \ + --output_dir path_to_ppo_checkpoint \ + --per_device_train_batch_size 2 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 1e-5 \ + --num_train_epochs 1.0 \ + --plot_loss +``` + +#### DPO 训练 + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage dpo \ + --model_name_or_path path_to_llama_model \ + --do_train \ + --dataset comparison_gpt4_zh \ + --template default \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --resume_lora_training False \ + --checkpoint_dir path_to_sft_checkpoint \ + --output_dir path_to_dpo_checkpoint \ + --per_device_train_batch_size 2 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 1e-5 \ + --num_train_epochs 1.0 \ + --plot_loss \ + --fp16 +``` + +### 多 GPU 分布式训练 + +#### 使用 Huggingface Accelerate + +```bash +accelerate config # 首先配置分布式环境 +accelerate launch src/train_bash.py # 参数同上 +``` + +
LoRA 训练的 Accelerate 配置示例 + +```yaml +compute_environment: LOCAL_MACHINE +distributed_type: MULTI_GPU +downcast_bf16: 'no' +gpu_ids: all +machine_rank: 0 +main_training_function: main +mixed_precision: fp16 +num_machines: 1 +num_processes: 4 +rdzv_backend: static +same_network: true +tpu_env: [] +tpu_use_cluster: false +tpu_use_sudo: false +use_cpu: false +``` + +
+ +#### 使用 DeepSpeed + +```bash +deepspeed --num_gpus 8 --master_port=9901 src/train_bash.py \ + --deepspeed ds_config.json \ + ... # 参数同上 +``` + +
使用 DeepSpeed ZeRO-2 进行全参数训练的 DeepSpeed 配置示例 + +```json +{ + "train_batch_size": "auto", + "train_micro_batch_size_per_gpu": "auto", + "gradient_accumulation_steps": "auto", + "gradient_clipping": "auto", + "zero_allow_untested_optimizer": true, + "fp16": { + "enabled": "auto", + "loss_scale": 0, + "initial_scale_power": 16, + "loss_scale_window": 1000, + "hysteresis": 2, + "min_loss_scale": 1 + }, + "zero_optimization": { + "stage": 2, + "allgather_partitions": true, + "allgather_bucket_size": 5e8, + "reduce_scatter": true, + "reduce_bucket_size": 5e8, + "overlap_comm": false, + "contiguous_gradients": true + } +} +``` + +
+ +### 导出微调后的完整模型 + +```bash +python src/export_model.py \ + --model_name_or_path path_to_llama_model \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint \ + --export_dir path_to_export +``` + +### API 服务 + +```bash +python src/api_demo.py \ + --model_name_or_path path_to_llama_model \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint +``` + +> [!NOTE] +> 关于 API 文档请见 `http://localhost:8000/docs`。 + +### 命令行测试 + +```bash +python src/cli_demo.py \ + --model_name_or_path path_to_llama_model \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint +``` + +### 浏览器测试 + +```bash +python src/web_demo.py \ + --model_name_or_path path_to_llama_model \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint +``` + +### 模型评估 + +```bash +CUDA_VISIBLE_DEVICES=0 python src/evaluate.py \ + --model_name_or_path path_to_llama_model \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint \ + --template vanilla \ + --task ceval \ + --split validation \ + --lang zh \ + --n_shot 5 \ + --batch_size 4 +``` + +### 模型预测 + +```bash +CUDA_VISIBLE_DEVICES=0 python src/train_bash.py \ + --stage sft \ + --model_name_or_path path_to_llama_model \ + --do_predict \ + --dataset alpaca_gpt4_zh \ + --template default \ + --finetuning_type lora \ + --checkpoint_dir path_to_checkpoint \ + --output_dir path_to_predict_result \ + --per_device_eval_batch_size 8 \ + --max_samples 100 \ + --predict_with_generate +``` + +> [!NOTE] +> 我们建议在量化模型的预测中使用 `--per_device_eval_batch_size=1` 和 `--max_target_length 128`。 + +## 使用了 LLaMA Factory 的项目 + +- **[StarWhisper](https://github.com/Yu-Yang-Li/StarWhisper)**: 天文大模型 StarWhisper,基于 ChatGLM2-6B 和 Qwen-14B 在天文数据上微调而得。 +- **[DISC-LawLLM](https://github.com/FudanDISC/DISC-LawLLM)**: 中文法律领域大模型 DISC-LawLLM,基于 Baichuan-13B 微调而得,具有法律推理和知识检索能力。 +- **[Sunsimiao](https://github.com/thomas-yanxin/Sunsimiao)**: 孙思邈中文医疗大模型 Sumsimiao,基于 Baichuan-7B 和 ChatGLM-6B 在中文医疗数据上微调而得。 +- **[CareGPT](https://github.com/WangRongsheng/CareGPT)**: 医疗大模型项目 CareGPT,基于 LLaMA2-7B 和 Baichuan-13B 在中文医疗数据上微调而得。 + +## 协议 + +本仓库的代码依照 [Apache-2.0](LICENSE) 协议开源。 + +使用模型权重时,请遵循对应的模型协议:[Baichuan](https://huggingface.co/baichuan-inc/Baichuan-13B-Base/resolve/main/Community%20License%20for%20Baichuan-13B%20Model.pdf) / [Baichuan2](https://huggingface.co/baichuan-inc/Baichuan2-13B-Chat/resolve/main/Community%20License%20for%20Baichuan2%20Model.pdf) / [BLOOM](https://huggingface.co/spaces/bigscience/license) / [ChatGLM3](https://github.com/THUDM/ChatGLM3/blob/main/MODEL_LICENSE) / [Falcon](https://huggingface.co/tiiuae/falcon-180B/blob/main/LICENSE.txt) / [InternLM](https://github.com/InternLM/InternLM#license) / [LLaMA](https://github.com/facebookresearch/llama/blob/main/MODEL_CARD.md) / [LLaMA-2](https://ai.meta.com/llama/license/) / [Mistral](LICENSE) / [Phi-1.5](https://huggingface.co/microsoft/phi-1_5/resolve/main/Research%20License.docx) / [Qwen](https://huggingface.co/Qwen/Qwen-7B-Chat/blob/main/LICENSE) / [XVERSE](https://github.com/xverse-ai/XVERSE-13B/blob/main/MODEL_LICENSE.pdf) + +## 引用 + +如果您觉得此项目有帮助,请考虑以下列格式引用 + +```bibtex +@Misc{llama-factory, + title = {LLaMA Factory}, + author = {hiyouga}, + howpublished = {\url{https://github.com/hiyouga/LLaMA-Factory}}, + year = {2023} +} +``` + +## 致谢 + +本项目受益于 [PEFT](https://github.com/huggingface/peft)、[QLoRA](https://github.com/artidoro/qlora) 和 [FastChat](https://github.com/lm-sys/FastChat),感谢以上诸位作者的付出。 + +## Star History + +![Star History Chart](https://api.star-history.com/svg?repos=hiyouga/LLaMA-Factory&type=Date) diff --git a/llm_rl/assets/wechat.jpg b/llm_rl/assets/wechat.jpg new file mode 100644 index 0000000000000000000000000000000000000000..8df68f9cbab8eda4a380b4f866adb0a569505aec GIT binary patch literal 143444 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.0"] +build-backend = "setuptools.build_meta" diff --git a/llm_rl/requirements.txt b/llm_rl/requirements.txt new file mode 100644 index 00000000..840d2f2d --- /dev/null +++ b/llm_rl/requirements.txt @@ -0,0 +1,20 @@ +torch>=1.13.1 +transformers>=4.31.0,<4.35.0 +datasets>=2.12.0 +accelerate>=0.21.0 +peft>=0.4.0 +trl>=0.7.2 +gradio>=3.38.0,<4.0.0 +scipy +sentencepiece +protobuf +tiktoken +fire +jieba +rouge-chinese +nltk +uvicorn +pydantic +fastapi +sse-starlette +matplotlib diff --git a/llm_rl/reward_model.sh b/llm_rl/reward_model.sh new file mode 100644 index 00000000..3068fb43 --- /dev/null +++ b/llm_rl/reward_model.sh @@ -0,0 +1,21 @@ +python src/train_bash.py \ + --stage rm \ + --model_name_or_path meta-llama/Llama-2-13b \ + --do_train \ + --dataset comparison_gpt4_en \ + --template default \ + --finetuning_type lora \ + --lora_target q_proj,v_proj \ + --resume_lora_training False \ + --checkpoint_dir ./llama-2-13b-rm \ + --output_dir ./llama-2-13b-rm \ + --per_device_train_batch_size 2 \ + --gradient_accumulation_steps 4 \ + --lr_scheduler_type cosine \ + --logging_steps 10 \ + --save_steps 1000 \ + --learning_rate 1e-6 \ + --num_train_epochs 1.0 \ + --plot_loss \ + --fp16 \ + --hf_auth_token "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" \ No newline at end of file diff --git a/llm_rl/setup.py b/llm_rl/setup.py new file mode 100644 index 00000000..7638eaab --- /dev/null +++ b/llm_rl/setup.py @@ -0,0 +1,55 @@ +import os +import re +from setuptools import setup, find_packages + + +def get_version(): + with open(os.path.join("src", "llmtuner", "__init__.py"), "r", encoding="utf-8") as f: + file_content = f.read() + pattern = r"{0}\W*=\W*\"([^\"]+)\"".format("__version__") + version, = re.findall(pattern, file_content) + return version + + +def get_requires(): + with open("requirements.txt", "r", encoding="utf-8") as f: + file_content = f.read() + lines = [line.strip() for line in file_content.strip().split("\n") if not line.startswith("#")] + return lines + + +def main(): + + setup( + name="llmtuner", + version=get_version(), + author="hiyouga", + author_email="hiyouga" "@" "buaa.edu.cn", + description="Easy-to-use LLM fine-tuning framework", + long_description=open("README.md", "r", encoding="utf-8").read(), + long_description_content_type="text/markdown", + keywords=["LLaMA", "BLOOM", "Falcon", "LLM", "ChatGPT", "transformer", "pytorch", "deep learning"], + license="Apache 2.0 License", + url="https://github.com/hiyouga/LLaMA-Factory", + package_dir={"": "src"}, + packages=find_packages("src"), + python_requires=">=3.8.0", + install_requires=get_requires(), + classifiers=[ + "Development Status :: 3 - Alpha", + "Intended Audience :: Developers", + "Intended Audience :: Education", + "Intended Audience :: Science/Research", + "License :: OSI Approved :: Apache Software License", + "Operating System :: OS Independent", + "Programming Language :: Python :: 3", + "Programming Language :: Python :: 3.8", + "Programming Language :: Python :: 3.9", + "Programming Language :: Python :: 3.10", + "Topic :: Scientific/Engineering :: Artificial Intelligence", + ] + ) + + +if __name__ == "__main__": + main() diff --git a/llm_rl/src/api_demo.py b/llm_rl/src/api_demo.py new file mode 100644 index 00000000..720089fd --- /dev/null +++ b/llm_rl/src/api_demo.py @@ -0,0 +1,14 @@ +import uvicorn + +from llmtuner import ChatModel, create_app + + +def main(): + chat_model = ChatModel() + app = create_app(chat_model) + print("Visit http://localhost:8000/docs for API document.") + uvicorn.run(app, host="0.0.0.0", port=8000, workers=1) + + +if __name__ == "__main__": + main() diff --git a/llm_rl/src/cli_demo.py b/llm_rl/src/cli_demo.py new file mode 100644 index 00000000..fe6a0bc4 --- /dev/null +++ b/llm_rl/src/cli_demo.py @@ -0,0 +1,39 @@ +import readline +from llmtuner import ChatModel + + +def main(): + chat_model = ChatModel() + history = [] + print("Welcome to the CLI application, use `clear` to remove the history, use `exit` to exit the application.") + + while True: + try: + query = input("\nUser: ") + except UnicodeDecodeError: + print("Detected decoding error at the inputs, please set the terminal encoding to utf-8.") + continue + except Exception: + raise + + if query.strip() == "exit": + break + + if query.strip() == "clear": + history = [] + print("History has been removed.") + continue + + print("Assistant: ", end="", flush=True) + + response = "" + for new_text in chat_model.stream_chat(query, history): + print(new_text, end="", flush=True) + response += new_text + print() + + history = history + [(query, response)] + + +if __name__ == "__main__": + main() diff --git a/llm_rl/src/evaluate.py b/llm_rl/src/evaluate.py new file mode 100644 index 00000000..8af8c12c --- /dev/null +++ b/llm_rl/src/evaluate.py @@ -0,0 +1,190 @@ +# coding=utf-8 +# Evaluates the performance of pre-trained models. +# Usage: python evaluate.py --model_name_or_path path_to_model --checkpoint_dir path_to_ckpt --template vanilla +# --task ceval --split validation --lang zh --n_shot 5 --batch_size 4 --save_name result +# Inspired by: https://github.com/hendrycks/test/blob/master/evaluate_flan.py + +import os +import fire +import json +import torch +import numpy as np +import transformers +from collections import Counter +from datasets import load_dataset +from dataclasses import dataclass +from tqdm import tqdm, trange +from typing import TYPE_CHECKING, Dict, List, Literal, Optional, Tuple + +from llmtuner import ChatModel + +if TYPE_CHECKING: + from datasets import Dataset + + +choices = ["A", "B", "C", "D"] + + +@dataclass +class EvalTemplate: + + system: str + choice: str + answer: str + prefix: str + + def parse_example( + self, + example: Dict[str, str] + ) -> Tuple[str, str]: + candidates = [self.choice.format(choice=ch, content=example[ch]) for ch in choices if ch in example] + return "".join([example["question"]] + candidates + [self.answer]), example["answer"] + + def format_example( + self, + target_data: Dict[str, str], + support_set: "Dataset", + subject_name: str, + use_history: bool + ) -> Tuple[str, str, List[Tuple[str, str]]]: + query, resp = self.parse_example(target_data) + history = [self.parse_example(support_set[k]) for k in range(len(support_set))] + + if len(history): + temp = history.pop(0) + history.insert(0, (self.system.format(subject=subject_name) + temp[0], temp[1])) + else: + query = self.system.format(subject=subject_name) + query + + if not use_history: + query = "\n\n".join(["".join(item) for item in history] + [query]) + history = [] + return query.strip(), resp, history + + +eval_templates = { + "en": EvalTemplate( + system="The following are multiple choice questions (with answers) about {subject}.\n\n", + choice="\n{choice}. {content}", + answer="\nAnswer: ", + prefix=" " + ), + "zh": EvalTemplate( + system="以下是中国关于{subject}考试的单项选择题,请选出其中的正确答案。\n\n", + choice="\n{choice}. {content}", + answer="\n答案:", + prefix="\n" + ) +} + + +@torch.inference_mode() +def batch_inference( + chat_model: ChatModel, + batch_input: Dict[str, torch.Tensor], + prefix_char: str +) -> List[str]: + logits = chat_model.model(**batch_input).logits + lengths = torch.sum(batch_input["attention_mask"], dim=-1) + nextword_logits = torch.stack([logits[i, lengths[i] - 1] for i in range(len(lengths))], dim=0) + probs = torch.nn.functional.softmax( + torch.stack( + [ + nextword_logits[:, chat_model.tokenizer.encode(prefix_char + choice, add_special_tokens=False)[-1]] + for choice in choices + ], + dim=-1 + ), + dim=-1 + ).detach() + return [chr(ord("A") + offset.item()) for offset in torch.argmax(probs, dim=-1)] + + +def evaluate( + model_name_or_path: str, + finetuning_type: Optional[str] = "lora", + checkpoint_dir: Optional[str] = None, + template: Optional[str] = "vanilla", + task: Optional[str] = "ceval", + dataset_dir: Optional[str] = "evaluation", + split: Optional[Literal["validation", "test"]] = "validation", + lang: Optional[Literal["zh", "en"]] = "zh", + n_shot: Optional[int] = 5, + n_avg: Optional[int] = 1, + batch_size: Optional[int] = 4, + save_name: Optional[str] = None, + seed: Optional[int] = 42 +): + with open(os.path.join(dataset_dir, task, "mapping.json"), "r", encoding="utf-8") as f: + categorys: Dict[str, Dict[str, str]] = json.load(f) + + transformers.set_seed(seed) + chat_model = ChatModel(dict( + model_name_or_path=model_name_or_path, + finetuning_type=finetuning_type, + checkpoint_dir=checkpoint_dir, + template=template + )) + chat_model.tokenizer.padding_side = "right" # avoid overflow issue in batched inference for llama2 + eval_template = eval_templates[lang] + + category_corrects: Dict[str, np.ndarray] = { + subj: np.array([], dtype="bool") for subj in ["Average", "STEM", "Social Sciences", "Humanities", "Other"] + } + pbar = tqdm(categorys.keys(), desc="Processing subjects", position=0) + results = {} + for subject in pbar: + dataset = load_dataset(os.path.join(dataset_dir, task), subject) + labels, answers, all_outputs = [], [], [] + for epoch in range(n_avg): + pbar.set_postfix_str("{} Trial: {}".format(categorys[subject]["name"], epoch)) + inputs, outputs = [], [] + for i in trange(len(dataset[split]), desc="Formatting batches", position=1, leave=False): + support_set = dataset["train"].shuffle().select(range(min(n_shot, len(dataset["train"])))) + query, resp, history = eval_template.format_example( + target_data=dataset[split][i], + support_set=support_set, + subject_name=categorys[subject]["name"], + use_history=chat_model.template.use_history + ) + input_ids, _ = chat_model.template.encode_oneturn( + tokenizer=chat_model.tokenizer, query=query, resp=resp, history=history + ) + inputs.append({"input_ids": input_ids, "attention_mask": [1] * len(input_ids)}) + if epoch == 0: + labels.append(resp) + + for i in trange(0, len(inputs), batch_size, desc="Predicting batches", position=1, leave=False): + batch_input = chat_model.tokenizer.pad( + inputs[i : i + batch_size], return_attention_mask=True, return_tensors="pt" + ).to(chat_model.model.device) + preds = batch_inference(chat_model, batch_input, eval_template.prefix) + outputs += preds + all_outputs.append(outputs) + + for i in range(len(all_outputs[0])): + count = Counter([all_outputs[epoch][i] for epoch in range(n_avg)]) + answers.append(count.most_common(1)[0][0]) + + corrects = (np.array(answers) == np.array(labels)) + category_name = categorys[subject]["category"] + category_corrects[category_name] = np.concatenate([category_corrects[category_name], corrects], axis=0) + category_corrects["Average"] = np.concatenate([category_corrects["Average"], corrects], axis=0) + results[subject] = {str(i): answers[i] for i in range(len(answers))} + + score_info = "\n".join([ + "{:>15}: {:.2f}".format(category_name, 100 * np.mean(category_correct)) + for category_name, category_correct in category_corrects.items() if len(category_correct) + ]) + + print(score_info) + if save_name is not None: + with open(save_name + ".json", "w", encoding="utf-8", newline="\n") as f: + json.dump(results, f, indent=2) + + with open(save_name + ".log", "w", encoding="utf-8", newline="\n") as f: + f.write(score_info) + + +if __name__ == "__main__": + fire.Fire(evaluate) diff --git a/llm_rl/src/export_model.py b/llm_rl/src/export_model.py new file mode 100644 index 00000000..4baeb2c3 --- /dev/null +++ b/llm_rl/src/export_model.py @@ -0,0 +1,9 @@ +from llmtuner import export_model + + +def main(): + export_model() + + +if __name__ == "__main__": + main() diff --git a/llm_rl/src/llmtuner/__init__.py b/llm_rl/src/llmtuner/__init__.py new file mode 100644 index 00000000..37eb9535 --- /dev/null +++ b/llm_rl/src/llmtuner/__init__.py @@ -0,0 +1,9 @@ +# Level: api, webui > chat > tuner > dsets > extras, hparams + +from llmtuner.api import create_app +from llmtuner.chat import ChatModel +from llmtuner.tuner import export_model, run_exp +from llmtuner.webui import create_ui, create_web_demo + + +__version__ = "0.2.0" diff --git a/llm_rl/src/llmtuner/api/__init__.py b/llm_rl/src/llmtuner/api/__init__.py new file mode 100644 index 00000000..b3ce183a --- /dev/null +++ b/llm_rl/src/llmtuner/api/__init__.py @@ -0,0 +1 @@ +from llmtuner.api.app import create_app diff --git a/llm_rl/src/llmtuner/api/app.py b/llm_rl/src/llmtuner/api/app.py new file mode 100644 index 00000000..27fb19e0 --- /dev/null +++ b/llm_rl/src/llmtuner/api/app.py @@ -0,0 +1,146 @@ +import json +import uvicorn +from fastapi import FastAPI, HTTPException, status +from fastapi.middleware.cors import CORSMiddleware +from contextlib import asynccontextmanager +from sse_starlette import EventSourceResponse +from typing import List, Tuple +from pydantic import BaseModel + +from llmtuner.extras.misc import torch_gc +from llmtuner.chat import ChatModel +from llmtuner.api.protocol import ( + Role, + Finish, + ModelCard, + ModelList, + ChatMessage, + DeltaMessage, + ChatCompletionRequest, + ChatCompletionResponse, + ChatCompletionStreamResponse, + ChatCompletionResponseChoice, + ChatCompletionResponseStreamChoice, + ChatCompletionResponseUsage +) + + +@asynccontextmanager +async def lifespan(app: FastAPI): # collects GPU memory + yield + torch_gc() + + +def to_json(data: BaseModel) -> str: + try: # pydantic v2 + return json.dumps(data.model_dump(exclude_unset=True), ensure_ascii=False) + except: # pydantic v1 + return data.json(exclude_unset=True, ensure_ascii=False) + + +def create_app(chat_model: ChatModel) -> FastAPI: + app = FastAPI(lifespan=lifespan) + + app.add_middleware( + CORSMiddleware, + allow_origins=["*"], + allow_credentials=True, + allow_methods=["*"], + allow_headers=["*"], + ) + + @app.get("/v1/models", response_model=ModelList) + async def list_models(): + model_card = ModelCard(id="gpt-3.5-turbo") + return ModelList(data=[model_card]) + + @app.post("/v1/chat/completions", response_model=ChatCompletionResponse, status_code=status.HTTP_200_OK) + async def create_chat_completion(request: ChatCompletionRequest): + if len(request.messages) < 1 or request.messages[-1].role != Role.USER: + raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="Invalid request") + + query = request.messages[-1].content + prev_messages = request.messages[:-1] + if len(prev_messages) > 0 and prev_messages[0].role == Role.SYSTEM: + system = prev_messages.pop(0).content + else: + system = None + + history = [] + if len(prev_messages) % 2 == 0: + for i in range(0, len(prev_messages), 2): + if prev_messages[i].role == Role.USER and prev_messages[i+1].role == Role.ASSISTANT: + history.append([prev_messages[i].content, prev_messages[i+1].content]) + else: + raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail="Only supports u/a/u/a/u...") + + if request.stream: + generate = predict(query, history, system, request) + return EventSourceResponse(generate, media_type="text/event-stream") + + response, (prompt_length, response_length) = chat_model.chat( + query, history, system, + do_sample=request.do_sample, + temperature=request.temperature, + top_p=request.top_p, + max_new_tokens=request.max_tokens, + num_return_sequences=request.n + ) + + usage = ChatCompletionResponseUsage( + prompt_tokens=prompt_length, + completion_tokens=response_length, + total_tokens=prompt_length+response_length + ) + + choices = [ChatCompletionResponseChoice( + index=i, + message=ChatMessage(role=Role.ASSISTANT, content=choice), + finish_reason=Finish.STOP + ) for i, choice in enumerate(response)] + + return ChatCompletionResponse(model=request.model, choices=choices, usage=usage) + + async def predict(query: str, history: List[Tuple[str, str]], system: str, request: ChatCompletionRequest): + choice_data = ChatCompletionResponseStreamChoice( + index=0, + delta=DeltaMessage(role=Role.ASSISTANT), + finish_reason=None + ) + chunk = ChatCompletionStreamResponse(model=request.model, choices=[choice_data]) + yield to_json(chunk) + + for new_text in chat_model.stream_chat( + query, history, system, + do_sample=request.do_sample, + temperature=request.temperature, + top_p=request.top_p, + max_new_tokens=request.max_tokens + ): + if len(new_text) == 0: + continue + + choice_data = ChatCompletionResponseStreamChoice( + index=0, + delta=DeltaMessage(content=new_text), + finish_reason=None + ) + chunk = ChatCompletionStreamResponse(model=request.model, choices=[choice_data]) + yield to_json(chunk) + + choice_data = ChatCompletionResponseStreamChoice( + index=0, + delta=DeltaMessage(), + finish_reason=Finish.STOP + ) + chunk = ChatCompletionStreamResponse(model=request.model, choices=[choice_data]) + yield to_json(chunk) + yield "[DONE]" + + return app + + +if __name__ == "__main__": + chat_model = ChatModel() + app = create_app(chat_model) + uvicorn.run(app, host="0.0.0.0", port=8000, workers=1) diff --git a/llm_rl/src/llmtuner/api/protocol.py b/llm_rl/src/llmtuner/api/protocol.py new file mode 100644 index 00000000..6b99da40 --- /dev/null +++ b/llm_rl/src/llmtuner/api/protocol.py @@ -0,0 +1,83 @@ +import time +from enum import Enum +from pydantic import BaseModel, Field +from typing import List, Optional + + +class Role(str, Enum): + USER = "user" + ASSISTANT = "assistant" + SYSTEM = "system" + + +class Finish(str, Enum): + STOP = "stop" + LENGTH = "length" + + +class ModelCard(BaseModel): + id: str + object: Optional[str] = "model" + created: Optional[int] = Field(default_factory=lambda: int(time.time())) + owned_by: Optional[str] = "owner" + + +class ModelList(BaseModel): + object: Optional[str] = "list" + data: Optional[List[ModelCard]] = [] + + +class ChatMessage(BaseModel): + role: Role + content: str + + +class DeltaMessage(BaseModel): + role: Optional[Role] = None + content: Optional[str] = None + + +class ChatCompletionRequest(BaseModel): + model: str + messages: List[ChatMessage] + do_sample: Optional[bool] = True + temperature: Optional[float] = None + top_p: Optional[float] = None + n: Optional[int] = 1 + max_tokens: Optional[int] = None + stream: Optional[bool] = False + + +class ChatCompletionResponseChoice(BaseModel): + index: int + message: ChatMessage + finish_reason: Finish + + +class ChatCompletionResponseStreamChoice(BaseModel): + index: int + delta: DeltaMessage + finish_reason: Optional[Finish] = None + + +class ChatCompletionResponseUsage(BaseModel): + prompt_tokens: int + completion_tokens: int + total_tokens: int + + +class ChatCompletionResponse(BaseModel): + id: Optional[str] = "chatcmpl-default" + object: Optional[str] = "chat.completion" + created: Optional[int] = Field(default_factory=lambda: int(time.time())) + model: str + choices: List[ChatCompletionResponseChoice] + usage: ChatCompletionResponseUsage + + +class ChatCompletionStreamResponse(BaseModel): + id: Optional[str] = "chatcmpl-default" + object: Optional[str] = "chat.completion.chunk" + created: Optional[int] = Field(default_factory=lambda: int(time.time())) + model: str + choices: List[ChatCompletionResponseStreamChoice] diff --git a/llm_rl/src/llmtuner/chat/__init__.py b/llm_rl/src/llmtuner/chat/__init__.py new file mode 100644 index 00000000..ba240d05 --- /dev/null +++ b/llm_rl/src/llmtuner/chat/__init__.py @@ -0,0 +1 @@ +from llmtuner.chat.stream_chat import ChatModel diff --git a/llm_rl/src/llmtuner/chat/stream_chat.py b/llm_rl/src/llmtuner/chat/stream_chat.py new file mode 100644 index 00000000..cc815d1b --- /dev/null +++ b/llm_rl/src/llmtuner/chat/stream_chat.py @@ -0,0 +1,109 @@ +import torch +from typing import Any, Dict, Generator, List, Optional, Tuple +from threading import Thread +from transformers import GenerationConfig, TextIteratorStreamer + +from llmtuner.extras.misc import dispatch_model, get_logits_processor +from llmtuner.extras.template import get_template_and_fix_tokenizer +from llmtuner.tuner.core import get_infer_args, load_model_and_tokenizer + + +class ChatModel: + + def __init__(self, args: Optional[Dict[str, Any]] = None) -> None: + model_args, data_args, finetuning_args, self.generating_args = get_infer_args(args) + self.model, self.tokenizer = load_model_and_tokenizer(model_args, finetuning_args) + self.tokenizer.padding_side = "left" + self.model = dispatch_model(self.model) + self.template = get_template_and_fix_tokenizer(data_args.template, self.tokenizer) + self.system_prompt = data_args.system_prompt + + def process_args( + self, + query: str, + history: Optional[List[Tuple[str, str]]] = None, + system: Optional[str] = None, + **input_kwargs + ) -> Tuple[Dict[str, Any], int]: + system = system or self.system_prompt + prompt, _ = self.template.encode_oneturn( + tokenizer=self.tokenizer, query=query, resp="", history=history, system=system + ) + prompt_length = len(prompt) + input_ids = torch.tensor([prompt], device=self.model.device) + + do_sample = input_kwargs.pop("do_sample", None) + temperature = input_kwargs.pop("temperature", None) + top_p = input_kwargs.pop("top_p", None) + top_k = input_kwargs.pop("top_k", None) + num_return_sequences = input_kwargs.pop("num_return_sequences", None) + repetition_penalty = input_kwargs.pop("repetition_penalty", None) + max_length = input_kwargs.pop("max_length", None) + max_new_tokens = input_kwargs.pop("max_new_tokens", None) + + generating_args = self.generating_args.to_dict() + generating_args.update(dict( + do_sample=do_sample if do_sample is not None else generating_args["do_sample"], + temperature=temperature or generating_args["temperature"], + top_p=top_p or generating_args["top_p"], + top_k=top_k or generating_args["top_k"], + num_return_sequences=num_return_sequences or 1, + repetition_penalty=repetition_penalty or generating_args["repetition_penalty"], + eos_token_id=[self.tokenizer.eos_token_id] + self.tokenizer.additional_special_tokens_ids, + pad_token_id=self.tokenizer.pad_token_id + )) + + if isinstance(num_return_sequences, int) and num_return_sequences > 1: + generating_args["do_sample"] = True + + if max_length: + generating_args.pop("max_new_tokens", None) + generating_args["max_length"] = max_length + + if max_new_tokens: + generating_args.pop("max_length", None) + generating_args["max_new_tokens"] = max_new_tokens + + gen_kwargs = dict( + inputs=input_ids, + generation_config=GenerationConfig(**generating_args), + logits_processor=get_logits_processor() + ) + + return gen_kwargs, prompt_length + + @torch.inference_mode() + def chat( + self, + query: str, + history: Optional[List[Tuple[str, str]]] = None, + system: Optional[str] = None, + **input_kwargs + ) -> Tuple[List[str], Tuple[int, int]]: + gen_kwargs, prompt_length = self.process_args(query, history, system, **input_kwargs) + generate_output = self.model.generate(**gen_kwargs) + response_ids = generate_output[:, prompt_length:] + response = self.tokenizer.batch_decode(response_ids, skip_special_tokens=True, clean_up_tokenization_spaces=True) + response_length = 0 + for i in range(len(response_ids)): + eos_index = (response_ids[i] == self.tokenizer.eos_token_id).nonzero() + response_length += eos_index[0].item() if len(eos_index) else len(response_ids[i]) + + return response, (prompt_length, response_length) + + @torch.inference_mode() + def stream_chat( + self, + query: str, + history: Optional[List[Tuple[str, str]]] = None, + system: Optional[str] = None, + **input_kwargs + ) -> Generator[str, None, None]: + gen_kwargs, _ = self.process_args(query, history, system, **input_kwargs) + streamer = TextIteratorStreamer(self.tokenizer, timeout=60.0, skip_prompt=True, skip_special_tokens=True) + gen_kwargs["streamer"] = streamer + + thread = Thread(target=self.model.generate, kwargs=gen_kwargs) + thread.start() + + yield from streamer diff --git a/llm_rl/src/llmtuner/dsets/__init__.py b/llm_rl/src/llmtuner/dsets/__init__.py new file mode 100644 index 00000000..cccbd745 --- /dev/null +++ b/llm_rl/src/llmtuner/dsets/__init__.py @@ -0,0 +1,3 @@ +from llmtuner.dsets.loader import get_dataset +from llmtuner.dsets.preprocess import preprocess_dataset +from llmtuner.dsets.utils import split_dataset diff --git a/llm_rl/src/llmtuner/dsets/loader.py b/llm_rl/src/llmtuner/dsets/loader.py new file mode 100644 index 00000000..834ef733 --- /dev/null +++ b/llm_rl/src/llmtuner/dsets/loader.py @@ -0,0 +1,145 @@ +import os +from typing import TYPE_CHECKING, Any, Dict, List, Union + +from datasets import concatenate_datasets, interleave_datasets, load_dataset + +from llmtuner.dsets.utils import checksum, EXT2TYPE +from llmtuner.extras.logging import get_logger + +if TYPE_CHECKING: + from datasets import Dataset, IterableDataset + from llmtuner.hparams import ModelArguments, DataArguments + + +logger = get_logger(__name__) + + +def get_dataset( + model_args: "ModelArguments", + data_args: "DataArguments" +) -> Union["Dataset", "IterableDataset"]: + max_samples = data_args.max_samples + all_datasets: List[Union["Dataset", "IterableDataset"]] = [] # support multiple datasets + + for dataset_attr in data_args.dataset_list: + logger.info("Loading dataset {}...".format(dataset_attr)) + + if dataset_attr.load_from == "hf_hub": + data_path = dataset_attr.dataset_name + data_name = dataset_attr.subset + data_files = None + elif dataset_attr.load_from == "script": + data_path = os.path.join(data_args.dataset_dir, dataset_attr.dataset_name) + data_name = dataset_attr.subset + data_files = None + elif dataset_attr.load_from == "file": + data_path, data_name = None, None + data_files: List[str] = [] + if os.path.isdir(os.path.join(data_args.dataset_dir, dataset_attr.dataset_name)): # is directory + for file_name in os.listdir(os.path.join(data_args.dataset_dir, dataset_attr.dataset_name)): + data_files.append(os.path.join(data_args.dataset_dir, dataset_attr.dataset_name, file_name)) + if data_path is None: + data_path = EXT2TYPE.get(file_name.split(".")[-1], None) + else: + assert data_path == EXT2TYPE.get(file_name.split(".")[-1], None), "file types are not identical." + elif os.path.isfile(os.path.join(data_args.dataset_dir, dataset_attr.dataset_name)): # is file + data_files.append(os.path.join(data_args.dataset_dir, dataset_attr.dataset_name)) + data_path = EXT2TYPE.get(dataset_attr.dataset_name.split(".")[-1], None) + else: + raise ValueError("File not found.") + + assert data_path, "File extension must be txt, csv, json or jsonl." + checksum(data_files, dataset_attr.dataset_sha1) + else: + raise NotImplementedError + + dataset = load_dataset( + path=data_path, + name=data_name, + data_files=data_files, + split=data_args.split, + cache_dir=model_args.cache_dir, + streaming=data_args.streaming, + use_auth_token=True if model_args.use_auth_token else None + ) + + if max_samples is not None: # truncate dataset + dataset = dataset.select(range(min(len(dataset), max_samples))) + + def convert_format(examples: Dict[str, List[Any]]) -> Dict[str, List[Any]]: + # convert dataset from sharegpt format to alpaca format + outputs = {"prompt": [], "query": [], "response": [], "history": []} + for msg_list in examples[dataset_attr.messages]: + msg_list = msg_list[:len(msg_list) // 2 * 2] # should be multiples of 2 + if len(msg_list) == 0: + continue + + msg_pairs = [] + user_role, assistant_role = None, None + for idx in range(0, len(msg_list), 2): + if user_role is None and assistant_role is None: + user_role = msg_list[idx][dataset_attr.role] + assistant_role = msg_list[idx + 1][dataset_attr.role] + else: + if ( + msg_list[idx][dataset_attr.role] != user_role + or msg_list[idx+1][dataset_attr.role] != assistant_role + ): + raise ValueError("Only accepts conversation in u/a/u/a/u/a order.") + msg_pairs.append((msg_list[idx][dataset_attr.content], msg_list[idx + 1][dataset_attr.content])) + + if len(msg_pairs) != 0: + outputs["prompt"].append(msg_pairs[-1][0]) + outputs["query"].append("") + outputs["response"].append(msg_pairs[-1][1]) + outputs["history"].append(msg_pairs[:-1]) + + return outputs + + if dataset_attr.formatting == "sharegpt": # convert format + column_names = list(next(iter(dataset)).keys()) + kwargs = {} + if not data_args.streaming: + kwargs = dict( + num_proc=data_args.preprocessing_num_workers, + load_from_cache_file=(not data_args.overwrite_cache), + desc="Converting format of dataset" + ) + + dataset = dataset.map( + convert_format, + batched=True, + remove_columns=column_names, + **kwargs + ) + else: + for column_name in ["prompt", "query", "response", "history"]: # align dataset + if getattr(dataset_attr, column_name) and getattr(dataset_attr, column_name) != column_name: + dataset = dataset.rename_column(getattr(dataset_attr, column_name), column_name) + + if dataset_attr.system_prompt: # add system prompt + system_prompt = dataset_attr.system_prompt + if data_args.streaming: + dataset = dataset.map(lambda _: {"system": system_prompt}) + else: + dataset = dataset.add_column("system", [system_prompt] * len(dataset)) + + all_datasets.append(dataset) + + if len(data_args.dataset_list) == 1: + return all_datasets[0] + elif data_args.mix_strategy == "concat": + if data_args.streaming: + logger.warning("The samples between different datasets will not be mixed in streaming mode.") + return concatenate_datasets(all_datasets) + elif data_args.mix_strategy.startswith("interleave"): + if not data_args.streaming: + logger.warning("We recommend using `mix_strategy=concat` in non-streaming mode.") + return interleave_datasets( + datasets=all_datasets, + probabilities=data_args.interleave_probs, + seed=data_args.seed, + stopping_strategy="first_exhausted" if data_args.mix_strategy.endswith("under") else "all_exhausted" + ) + else: + raise ValueError("Unknown mixing strategy.") diff --git a/llm_rl/src/llmtuner/dsets/preprocess.py b/llm_rl/src/llmtuner/dsets/preprocess.py new file mode 100644 index 00000000..0484b78e --- /dev/null +++ b/llm_rl/src/llmtuner/dsets/preprocess.py @@ -0,0 +1,268 @@ +import os +import tiktoken +from itertools import chain +from typing import TYPE_CHECKING, Any, Dict, Generator, List, Literal, Union + +from datasets import load_from_disk + +from llmtuner.extras.constants import IGNORE_INDEX +from llmtuner.extras.logging import get_logger +from llmtuner.extras.template import get_template_and_fix_tokenizer + +if TYPE_CHECKING: + from datasets import Dataset, IterableDataset + from transformers import Seq2SeqTrainingArguments + from transformers.tokenization_utils import PreTrainedTokenizer + from llmtuner.hparams import DataArguments + + +logger = get_logger(__name__) + + +def preprocess_dataset( + dataset: Union["Dataset", "IterableDataset"], + tokenizer: "PreTrainedTokenizer", + data_args: "DataArguments", + training_args: "Seq2SeqTrainingArguments", + stage: Literal["pt", "sft", "rm", "ppo"] +) -> Union["Dataset", "IterableDataset"]: + template = get_template_and_fix_tokenizer(data_args.template, tokenizer) + + if data_args.train_on_prompt and template.efficient_eos: + raise ValueError("Current template does not support `train_on_prompt`.") + + def construct_example(examples: Dict[str, List[Any]]) -> Generator[Any, None, None]: + for i in range(len(examples["prompt"])): + query, response = examples["prompt"][i], examples["response"][i] + query = query + "\n" + examples["query"][i] if "query" in examples and examples["query"][i] else query + history = examples["history"][i] if "history" in examples else None + system = examples["system"][i] if "system" in examples else None + yield query, response, history, system + + def preprocess_pretrain_dataset(examples: Dict[str, List[Any]]) -> Dict[str, List[List[int]]]: + # build grouped texts with format `X1 X2 X3 ...` + if isinstance(getattr(tokenizer, "tokenizer", None), tiktoken.Encoding): # for tiktoken tokenizer (Qwen) + kwargs = dict(allowed_special="all") + else: + kwargs = dict(add_special_tokens=True) + + if hasattr(tokenizer, "add_eos_token"): # for LLaMA tokenizer + setattr(tokenizer, "add_eos_token", True) + + tokenized_examples = tokenizer(examples["prompt"], **kwargs) + concatenated_examples = {k: list(chain(*tokenized_examples[k])) for k in tokenized_examples.keys()} + total_length = len(concatenated_examples[list(concatenated_examples.keys())[0]]) + block_size = data_args.cutoff_len + # we drop the small remainder, and if the total_length < block_size, we exclude this batch + total_length = (total_length // block_size) * block_size + # split by chunks of cutoff_len + result = { + k: [t[i: i + block_size] for i in range(0, total_length, block_size)] + for k, t in concatenated_examples.items() + } + return result + + def preprocess_supervised_dataset(examples: Dict[str, List[Any]]) -> Dict[str, List[List[int]]]: + # build inputs with format ` X Y ` and labels with format ` ... Y ` + # for multiturn examples, we only mask the prompt part in each prompt-response pair. + model_inputs = {"input_ids": [], "attention_mask": [], "labels": []} + + for query, response, history, system in construct_example(examples): + if not (isinstance(query, str) and isinstance(response, str) and query != "" and response != ""): + continue + + input_ids, labels = [], [] + for turn_idx, (source_ids, target_ids) in enumerate(template.encode_multiturn( + tokenizer, query, response, history, system + )): + total_len = len(source_ids) + len(target_ids) + max_source_len = int(data_args.cutoff_len * (len(source_ids) / total_len)) + max_target_len = int(data_args.cutoff_len * (len(target_ids) / total_len)) + + if len(source_ids) > max_source_len: + source_ids = source_ids[:max_source_len] + if len(target_ids) > max_target_len: + target_ids = target_ids[:max_target_len] + + if data_args.train_on_prompt: + source_mask = source_ids + elif turn_idx != 0 and template.efficient_eos: + source_mask = [tokenizer.eos_token_id] + [IGNORE_INDEX] * (len(source_ids) - 1) + else: + source_mask = [IGNORE_INDEX] * len(source_ids) + + input_ids += source_ids + target_ids + labels += source_mask + target_ids + + if template.efficient_eos: + input_ids += [tokenizer.eos_token_id] + labels += [tokenizer.eos_token_id] + + if len(input_ids) > data_args.cutoff_len: + input_ids = input_ids[:data_args.cutoff_len] + labels = labels[:data_args.cutoff_len] + + model_inputs["input_ids"].append(input_ids) + model_inputs["attention_mask"].append([1] * len(input_ids)) + model_inputs["labels"].append(labels) + + return model_inputs + + def preprocess_packed_supervised_dataset(examples: Dict[str, List[Any]]) -> Dict[str, List[List[int]]]: + # build inputs with format ` X1 Y1 X2 Y2 ` + # and labels with format ` ... Y1 ... Y2 ` + model_inputs = {"input_ids": [], "attention_mask": [], "labels": []} + input_ids, labels = [], [] + for query, response, history, system in construct_example(examples): + if not (isinstance(query, str) and isinstance(response, str) and query != "" and response != ""): + continue + + for turn_idx, (source_ids, target_ids) in enumerate(template.encode_multiturn( + tokenizer, query, response, history, system + )): + if data_args.train_on_prompt: + source_mask = source_ids + elif turn_idx != 0 and template.efficient_eos: + source_mask = [tokenizer.eos_token_id] + [IGNORE_INDEX] * (len(source_ids) - 1) + else: + source_mask = [IGNORE_INDEX] * len(source_ids) + input_ids += source_ids + target_ids + labels += source_mask + target_ids + + if template.efficient_eos: + input_ids += [tokenizer.eos_token_id] + labels += [tokenizer.eos_token_id] + + total_length = len(input_ids) + block_size = data_args.cutoff_len + # we drop the small remainder, and if the total_length < block_size, we exclude this batch + total_length = (total_length // block_size) * block_size + # split by chunks of cutoff_len + for i in range(0, total_length, block_size): + model_inputs["input_ids"].append(input_ids[i: i + block_size]) + model_inputs["attention_mask"].append([1] * block_size) + model_inputs["labels"].append(labels[i: i + block_size]) + + return model_inputs + + def preprocess_unsupervised_dataset(examples: Dict[str, List[Any]]) -> Dict[str, List[List[int]]]: + # build inputs with format ` X` and labels with format `Y ` + model_inputs = {"input_ids": [], "attention_mask": [], "labels": []} + + for query, response, history, system in construct_example(examples): + if not (isinstance(query, str) and query != ""): + continue + + input_ids, labels = template.encode_oneturn(tokenizer, query, response, history, system) + + if template.efficient_eos: + labels += [tokenizer.eos_token_id] + + if len(input_ids) > data_args.cutoff_len: + input_ids = input_ids[:data_args.cutoff_len] + if len(labels) > data_args.cutoff_len: + labels = labels[:data_args.cutoff_len] + + model_inputs["input_ids"].append(input_ids) + model_inputs["attention_mask"].append([1] * len(input_ids)) + model_inputs["labels"].append(labels) + + return model_inputs + + def preprocess_pairwise_dataset(examples: Dict[str, List[Any]]) -> Dict[str, List[List[int]]]: + # build input pairs with format ` X`, `Y1 ` and `Y2 ` + model_inputs = {"prompt_ids": [], "chosen_ids": [], "rejected_ids": []} + for query, response, history, system in construct_example(examples): + if not (isinstance(query, str) and isinstance(response, list) and query != "" and len(response) > 1): + continue + + prompt_ids, chosen_ids = template.encode_oneturn(tokenizer, query, response[0], history, system) + _, rejected_ids = template.encode_oneturn(tokenizer, query, response[1], history, system) + + if template.efficient_eos: + chosen_ids += [tokenizer.eos_token_id] + rejected_ids += [tokenizer.eos_token_id] + + total_len = len(prompt_ids) + max(len(chosen_ids), len(rejected_ids)) + max_source_len = int(data_args.cutoff_len * (len(prompt_ids) / total_len)) + max_target_len = int(data_args.cutoff_len * (max(len(chosen_ids), len(rejected_ids)) / total_len)) + + if len(prompt_ids) > max_source_len: + prompt_ids = prompt_ids[:max_source_len] + if len(chosen_ids) > max_target_len: + chosen_ids = chosen_ids[:max_target_len] + if len(rejected_ids) > max_target_len: + rejected_ids = rejected_ids[:max_target_len] + + model_inputs["prompt_ids"].append(prompt_ids) + model_inputs["chosen_ids"].append(chosen_ids) + model_inputs["rejected_ids"].append(rejected_ids) + + return model_inputs + + def print_supervised_dataset_example(example: Dict[str, List[int]]) -> None: + print("input_ids:\n{}".format(example["input_ids"])) + print("inputs:\n{}".format(tokenizer.decode(example["input_ids"], skip_special_tokens=False))) + print("label_ids:\n{}".format(example["labels"])) + print("labels:\n{}".format( + tokenizer.decode(list(filter(lambda x: x != IGNORE_INDEX, example["labels"])), skip_special_tokens=False) + )) + + def print_pairwise_dataset_example(example: Dict[str, List[int]]) -> None: + print("prompt_ids:\n{}".format(example["prompt_ids"])) + print("prompt:\n{}".format(tokenizer.decode(example["prompt_ids"], skip_special_tokens=False))) + print("chosen_ids:\n{}".format(example["chosen_ids"])) + print("chosen:\n{}".format(tokenizer.decode(example["chosen_ids"], skip_special_tokens=False))) + print("rejected_ids:\n{}".format(example["rejected_ids"])) + print("rejected:\n{}".format(tokenizer.decode(example["rejected_ids"], skip_special_tokens=False))) + + def print_unsupervised_dataset_example(example: Dict[str, List[int]]) -> None: + print("input_ids:\n{}".format(example["input_ids"])) + print("inputs:\n{}".format(tokenizer.decode(example["input_ids"], skip_special_tokens=False))) + + if stage == "pt": + preprocess_func = preprocess_pretrain_dataset + print_function = print_unsupervised_dataset_example + elif stage == "sft" and not training_args.predict_with_generate: + preprocess_func = preprocess_packed_supervised_dataset if data_args.sft_packing else preprocess_supervised_dataset + print_function = print_supervised_dataset_example + elif stage == "rm": + preprocess_func = preprocess_pairwise_dataset + print_function = print_pairwise_dataset_example + else: + preprocess_func = preprocess_unsupervised_dataset + print_function = print_unsupervised_dataset_example + + if data_args.cache_path is not None and os.path.exists(data_args.cache_path): + logger.warning("Loading dataset from disk will ignore other data arguments.") + return load_from_disk(data_args.cache_path) + + with training_args.main_process_first(desc="dataset map pre-processing"): + column_names = list(next(iter(dataset)).keys()) + kwargs = {} + if not data_args.streaming: + kwargs = dict( + num_proc=data_args.preprocessing_num_workers, + load_from_cache_file=(not data_args.overwrite_cache), + desc="Running tokenizer on dataset" + ) + + dataset = dataset.map( + preprocess_func, + batched=True, + remove_columns=column_names, + **kwargs + ) + + if data_args.cache_path is not None and not os.path.exists(data_args.cache_path): + if training_args.should_save: + dataset.save_to_disk(data_args.cache_path) + raise SystemExit("Dataset saved, rerun this script with the same `--cache_file`.") + + if training_args.should_log: + try: + print_function(next(iter(dataset))) + except StopIteration: + raise RuntimeError("Empty dataset!") + + return dataset diff --git a/llm_rl/src/llmtuner/dsets/utils.py b/llm_rl/src/llmtuner/dsets/utils.py new file mode 100644 index 00000000..bf337014 --- /dev/null +++ b/llm_rl/src/llmtuner/dsets/utils.py @@ -0,0 +1,59 @@ +import hashlib +from typing import TYPE_CHECKING, Dict, List, Optional, Union + +from llmtuner.extras.logging import get_logger + +if TYPE_CHECKING: + from datasets import Dataset, IterableDataset + from transformers import TrainingArguments + from llmtuner.hparams import DataArguments + + +logger = get_logger(__name__) + + +EXT2TYPE = { + "csv": "csv", + "json": "json", + "jsonl": "json", + "txt": "text" +} + + +def checksum(data_files: List[str], file_sha1: Optional[str] = None) -> None: + if file_sha1 is None: + logger.warning("Checksum failed: missing SHA-1 hash value in dataset_info.json.") + return + + if len(data_files) != 1: + logger.warning("Checksum failed: too many files.") + return + + with open(data_files[0], "rb") as f: + sha1 = hashlib.sha1(f.read()).hexdigest() + if sha1 != file_sha1: + logger.warning("Checksum failed: mismatched SHA-1 hash value at {}.".format(data_files[0])) + + +def split_dataset( + dataset: Union["Dataset", "IterableDataset"], + data_args: "DataArguments", + training_args: "TrainingArguments" +) -> Dict[str, "Dataset"]: + if training_args.do_train: + if data_args.val_size > 1e-6: # Split the dataset + if data_args.streaming: + val_set = dataset.take(int(data_args.val_size)) + train_set = dataset.skip(int(data_args.val_size)) + dataset = dataset.shuffle(buffer_size=data_args.buffer_size, seed=training_args.seed) + return {"train_dataset": train_set, "eval_dataset": val_set} + else: + val_size = int(data_args.val_size) if data_args.val_size > 1 else data_args.val_size + dataset = dataset.train_test_split(test_size=val_size, seed=training_args.seed) + return {"train_dataset": dataset["train"], "eval_dataset": dataset["test"]} + else: + if data_args.streaming: + dataset = dataset.shuffle(buffer_size=data_args.buffer_size, seed=training_args.seed) + return {"train_dataset": dataset} + else: # do_eval or do_predict + return {"eval_dataset": dataset} diff --git a/llm_rl/src/llmtuner/extras/__init__.py b/llm_rl/src/llmtuner/extras/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/llm_rl/src/llmtuner/extras/callbacks.py b/llm_rl/src/llmtuner/extras/callbacks.py new file mode 100644 index 00000000..7398d424 --- /dev/null +++ b/llm_rl/src/llmtuner/extras/callbacks.py @@ -0,0 +1,155 @@ +import os +import json +import time +from typing import TYPE_CHECKING +from datetime import timedelta + +from transformers import TrainerCallback +from transformers.trainer_utils import has_length, PREFIX_CHECKPOINT_DIR + +from llmtuner.extras.constants import LOG_FILE_NAME +from llmtuner.extras.logging import get_logger + +if TYPE_CHECKING: + from transformers import TrainingArguments, TrainerState, TrainerControl + + +logger = get_logger(__name__) + + +class SavePeftModelCallback(TrainerCallback): + + def on_save(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", **kwargs): + r""" + Event called after a checkpoint save. + """ + if args.should_save: + output_dir = os.path.join(args.output_dir, "{}-{}".format(PREFIX_CHECKPOINT_DIR, state.global_step)) + model = kwargs.pop("model") + if getattr(model, "is_peft_model", False): + getattr(model, "pretrained_model").save_pretrained(output_dir) + + def on_train_end(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", **kwargs): + r""" + Event called at the end of training. + """ + if args.should_save: + model = kwargs.pop("model") + if getattr(model, "is_peft_model", False): + getattr(model, "pretrained_model").save_pretrained(args.output_dir) + + +class LogCallback(TrainerCallback): + + def __init__(self, runner=None): + self.runner = runner + self.in_training = False + self.start_time = time.time() + self.cur_steps = 0 + self.max_steps = 0 + self.elapsed_time = "" + self.remaining_time = "" + + def timing(self): + cur_time = time.time() + elapsed_time = cur_time - self.start_time + avg_time_per_step = elapsed_time / self.cur_steps if self.cur_steps != 0 else 0 + remaining_time = (self.max_steps - self.cur_steps) * avg_time_per_step + self.elapsed_time = str(timedelta(seconds=int(elapsed_time))) + self.remaining_time = str(timedelta(seconds=int(remaining_time))) + + def on_train_begin(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", **kwargs): + r""" + Event called at the beginning of training. + """ + if state.is_local_process_zero: + self.in_training = True + self.start_time = time.time() + self.max_steps = state.max_steps + if os.path.exists(os.path.join(args.output_dir, LOG_FILE_NAME)) and args.overwrite_output_dir: + logger.warning("Previous log file in this folder will be deleted.") + os.remove(os.path.join(args.output_dir, LOG_FILE_NAME)) + + def on_train_end(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", **kwargs): + r""" + Event called at the end of training. + """ + if state.is_local_process_zero: + self.in_training = False + self.cur_steps = 0 + self.max_steps = 0 + + def on_substep_end(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", **kwargs): + r""" + Event called at the end of an substep during gradient accumulation. + """ + if state.is_local_process_zero and self.runner is not None and self.runner.aborted: + control.should_epoch_stop = True + control.should_training_stop = True + + def on_step_end(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", **kwargs): + r""" + Event called at the end of a training step. + """ + if state.is_local_process_zero: + self.cur_steps = state.global_step + self.timing() + if self.runner is not None and self.runner.aborted: + control.should_epoch_stop = True + control.should_training_stop = True + + def on_evaluate(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", **kwargs): + r""" + Event called after an evaluation phase. + """ + if state.is_local_process_zero and not self.in_training: + self.cur_steps = 0 + self.max_steps = 0 + + def on_predict(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", *other, **kwargs): + r""" + Event called after a successful prediction. + """ + if state.is_local_process_zero and not self.in_training: + self.cur_steps = 0 + self.max_steps = 0 + + def on_log(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", **kwargs) -> None: + r""" + Event called after logging the last logs. + """ + if not state.is_local_process_zero: + return + + logs = dict( + current_steps=self.cur_steps, + total_steps=self.max_steps, + loss=state.log_history[-1].get("loss", None), + eval_loss=state.log_history[-1].get("eval_loss", None), + predict_loss=state.log_history[-1].get("predict_loss", None), + reward=state.log_history[-1].get("reward", None), + learning_rate=state.log_history[-1].get("learning_rate", None), + epoch=state.log_history[-1].get("epoch", None), + percentage=round(self.cur_steps / self.max_steps * 100, 2) if self.max_steps != 0 else 100, + elapsed_time=self.elapsed_time, + remaining_time=self.remaining_time + ) + if self.runner is not None: + logger.info("{{'loss': {:.4f}, 'learning_rate': {:2.4e}, 'epoch': {:.2f}}}".format( + logs["loss"] or 0, logs["learning_rate"] or 0, logs["epoch"] or 0 + )) + + os.makedirs(args.output_dir, exist_ok=True) + with open(os.path.join(args.output_dir, "trainer_log.jsonl"), "a", encoding="utf-8") as f: + f.write(json.dumps(logs) + "\n") + + def on_prediction_step(self, args: "TrainingArguments", state: "TrainerState", control: "TrainerControl", **kwargs): + r""" + Event called after a prediction step. + """ + eval_dataloader = kwargs.pop("eval_dataloader", None) + if state.is_local_process_zero and has_length(eval_dataloader) and not self.in_training: + if self.max_steps == 0: + self.max_steps = len(eval_dataloader) + self.cur_steps += 1 + self.timing() diff --git a/llm_rl/src/llmtuner/extras/constants.py b/llm_rl/src/llmtuner/extras/constants.py new file mode 100644 index 00000000..dc55a080 --- /dev/null +++ b/llm_rl/src/llmtuner/extras/constants.py @@ -0,0 +1,92 @@ +IGNORE_INDEX = -100 + +LOG_FILE_NAME = "trainer_log.jsonl" + +LAYERNORM_NAMES = ["norm", "ln_f", "ln_attn", "ln_mlp", "ln_1", "ln_2"] + +METHODS = ["full", "freeze", "lora"] + +TRAINING_STAGES = { + "Supervised Fine-Tuning": "sft", + "Reward Modeling": "rm", + "PPO": "ppo", + "DPO": "dpo", + "Pre-Training": "pt" +} + +SUPPORTED_MODELS = { + "LLaMA-7B": "huggyllama/llama-7b", + "LLaMA-13B": "huggyllama/llama-13b", + "LLaMA-30B": "huggyllama/llama-30b", + "LLaMA-65B": "huggyllama/llama-65b", + "LLaMA2-7B": "meta-llama/Llama-2-7b-hf", + "LLaMA2-13B": "meta-llama/Llama-2-13b-hf", + "LLaMA2-70B": "meta-llama/Llama-2-70b-hf", + "LLaMA2-7B-Chat": "meta-llama/Llama-2-7b-chat-hf", + "LLaMA2-13B-Chat": "meta-llama/Llama-2-13b-chat-hf", + "LLaMA2-70B-Chat": "meta-llama/Llama-2-70b-chat-hf", + "ChineseLLaMA2-7B": "ziqingyang/chinese-llama-2-7b", + "ChineseLLaMA2-13B": "ziqingyang/chinese-llama-2-13b", + "ChineseLLaMA2-7B-Chat": "ziqingyang/chinese-alpaca-2-7b", + "ChineseLLaMA2-13B-Chat": "ziqingyang/chinese-alpaca-2-13b", + "BLOOM-560M": "bigscience/bloom-560m", + "BLOOM-3B": "bigscience/bloom-3b", + "BLOOM-7B1": "bigscience/bloom-7b1", + "BLOOMZ-560M": "bigscience/bloomz-560m", + "BLOOMZ-3B": "bigscience/bloomz-3b", + "BLOOMZ-7B1-mt": "bigscience/bloomz-7b1-mt", + "Falcon-7B": "tiiuae/falcon-7b", + "Falcon-40B": "tiiuae/falcon-40b", + "Falcon-7B-Chat": "tiiuae/falcon-7b-instruct", + "Falcon-40B-Chat": "tiiuae/falcon-40b-instruct", + "Baichuan-7B": "baichuan-inc/Baichuan-7B", + "Baichuan-13B": "baichuan-inc/Baichuan-13B-Base", + "Baichuan-13B-Chat": "baichuan-inc/Baichuan-13B-Chat", + "Baichuan2-7B": "baichuan-inc/Baichuan2-7B-Base", + "Baichuan2-13B": "baichuan-inc/Baichuan2-13B-Base", + "Baichuan2-7B-Chat": "baichuan-inc/Baichuan2-7B-Chat", + "Baichuan2-13B-Chat": "baichuan-inc/Baichuan2-13B-Chat", + "InternLM-7B": "internlm/internlm-7b", + "InternLM-20B": "internlm/internlm-20b", + "InternLM-7B-Chat": "internlm/internlm-chat-7b", + "InternLM-20B-Chat": "internlm/internlm-chat-20b", + "Qwen-7B": "Qwen/Qwen-7B", + "Qwen-14B": "Qwen/Qwen-14B", + "Qwen-7B-Chat": "Qwen/Qwen-7B-Chat", + "Qwen-14B-Chat": "Qwen/Qwen-14B-Chat", + "XVERSE-13B": "xverse/XVERSE-13B", + "XVERSE-13B-Chat": "xverse/XVERSE-13B-Chat", + "ChatGLM2-6B-Chat": "THUDM/chatglm2-6b", + "ChatGLM3-6B-Base": "THUDM/chatglm3-6b-base", + "ChatGLM3-6B-Chat": "THUDM/chatglm3-6b", + "Phi1.5-1.3B": "microsoft/phi-1_5" +} + +DEFAULT_MODULE = { + "LLaMA": "q_proj,v_proj", + "LLaMA2": "q_proj,v_proj", + "ChineseLLaMA2": "q_proj,v_proj", + "BLOOM": "query_key_value", + "BLOOMZ": "query_key_value", + "Falcon": "query_key_value", + "Baichuan": "W_pack", + "Baichuan2": "W_pack", + "InternLM": "q_proj,v_proj", + "Qwen": "c_attn", + "XVERSE": "q_proj,v_proj", + "ChatGLM2": "query_key_value", + "ChatGLM3": "query_key_value", + "Phi1.5": "Wqkv" +} + +DEFAULT_TEMPLATE = { + "LLaMA2": "llama2", + "ChineseLLaMA2": "llama2_zh", + "Baichuan": "baichuan", + "Baichuan2": "baichuan2", + "InternLM": "intern", + "Qwen": "chatml", + "XVERSE": "xverse", + "ChatGLM2": "chatglm2", + "ChatGLM3": "chatglm3" +} diff --git a/llm_rl/src/llmtuner/extras/logging.py b/llm_rl/src/llmtuner/extras/logging.py new file mode 100644 index 00000000..d6f185e6 --- /dev/null +++ b/llm_rl/src/llmtuner/extras/logging.py @@ -0,0 +1,43 @@ +import sys +import logging + + +class LoggerHandler(logging.Handler): + + def __init__(self): + super().__init__() + self.log = "" + + def reset(self): + self.log = "" + + def emit(self, record): + if record.name == "httpx": + return + log_entry = self.format(record) + self.log += log_entry + self.log += "\n\n" + + +def reset_logging(): + r""" + Removes basic config of root logger + """ + root = logging.getLogger() + list(map(root.removeHandler, root.handlers)) + list(map(root.removeFilter, root.filters)) + + +def get_logger(name: str) -> logging.Logger: + formatter = logging.Formatter( + fmt="%(asctime)s - %(levelname)s - %(name)s - %(message)s", + datefmt="%m/%d/%Y %H:%M:%S" + ) + handler = logging.StreamHandler(sys.stdout) + handler.setFormatter(formatter) + + logger = logging.getLogger(name) + logger.setLevel(logging.INFO) + logger.addHandler(handler) + + return logger diff --git a/llm_rl/src/llmtuner/extras/misc.py b/llm_rl/src/llmtuner/extras/misc.py new file mode 100644 index 00000000..960d43ee --- /dev/null +++ b/llm_rl/src/llmtuner/extras/misc.py @@ -0,0 +1,118 @@ +import gc +import torch +from typing import TYPE_CHECKING, Tuple +from transformers import InfNanRemoveLogitsProcessor, LogitsProcessorList + +try: + from transformers.utils import ( + is_torch_bf16_cpu_available, + is_torch_bf16_gpu_available, + is_torch_cuda_available, + is_torch_npu_available + ) + _is_fp16_available = is_torch_npu_available() or is_torch_cuda_available() + _is_bf16_available = is_torch_bf16_gpu_available() or is_torch_bf16_cpu_available +except ImportError: + _is_fp16_available = torch.cuda.is_available() + _is_bf16_available = torch.cuda.is_bf16_supported() + +if TYPE_CHECKING: + from transformers.modeling_utils import PreTrainedModel + + +class AverageMeter: + r""" + Computes and stores the average and current value. + """ + def __init__(self): + self.reset() + + def reset(self): + self.val = 0 + self.avg = 0 + self.sum = 0 + self.count = 0 + + def update(self, val, n=1): + self.val = val + self.sum += val * n + self.count += n + self.avg = self.sum / self.count + + +def count_parameters(model: torch.nn.Module) -> Tuple[int, int]: + r""" + Returns the number of trainable parameters and number of all parameters in the model. + """ + trainable_params, all_param = 0, 0 + for param in model.parameters(): + num_params = param.numel() + # if using DS Zero 3 and the weights are initialized empty + if num_params == 0 and hasattr(param, "ds_numel"): + num_params = param.ds_numel + + # Due to the design of 4bit linear layers from bitsandbytes, multiply the number of parameters by 2 + if param.__class__.__name__ == "Params4bit": + num_params = num_params * 2 + + all_param += num_params + if param.requires_grad: + trainable_params += num_params + + return trainable_params, all_param + + +def infer_optim_dtype(model_dtype: torch.dtype) -> torch.dtype: + r""" + Infers the optimal dtype according to the model_dtype and device compatibility. + """ + if _is_bf16_available and model_dtype == torch.bfloat16: + return torch.bfloat16 + elif _is_fp16_available: + return torch.float16 + else: + return torch.float32 + + +def get_logits_processor() -> LogitsProcessorList: + r""" + Gets logits processor that removes NaN and Inf logits. + """ + logits_processor = LogitsProcessorList() + logits_processor.append(InfNanRemoveLogitsProcessor()) + return logits_processor + + +def torch_gc() -> None: + r""" + Collects GPU memory. + """ + gc.collect() + if torch.cuda.is_available(): + torch.cuda.empty_cache() + torch.cuda.ipc_collect() + + +def dispatch_model(model: "PreTrainedModel") -> "PreTrainedModel": + r""" + Dispatches a pre-trained model to GPUs with balanced memory. + Borrowed from: https://github.com/huggingface/transformers/blob/v4.31.0/src/transformers/modeling_utils.py#L2803 + """ + if getattr(model, "is_loaded_in_8bit", False) or getattr(model, "is_loaded_in_4bit", False): # do nothing + return model + + if torch.cuda.device_count() > 1: + from accelerate import dispatch_model + from accelerate.utils import infer_auto_device_map, get_balanced_memory + + if model._no_split_modules is None: + raise ValueError("The model class needs to implement the `_no_split_modules` attribute.") + + kwargs = {"dtype": model.dtype, "no_split_module_classes": model._no_split_modules} + max_memory = get_balanced_memory(model, **kwargs) + # Make sure tied weights are tied before creating the device map. + model.tie_weights() + device_map = infer_auto_device_map(model, max_memory=max_memory, **kwargs) + return dispatch_model(model, device_map) + else: + return model.cuda() diff --git a/llm_rl/src/llmtuner/extras/patches/__init__.py b/llm_rl/src/llmtuner/extras/patches/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/llm_rl/src/llmtuner/extras/patches/llama_patch.py b/llm_rl/src/llmtuner/extras/patches/llama_patch.py new file mode 100644 index 00000000..a8473311 --- /dev/null +++ b/llm_rl/src/llmtuner/extras/patches/llama_patch.py @@ -0,0 +1,218 @@ +import math +import torch +import torch.nn as nn +from typing import Optional, Tuple +from transformers.utils import logging +from transformers.models.llama.modeling_llama import LlamaAttention, apply_rotary_pos_emb, repeat_kv + +try: + from flash_attn import flash_attn_func, flash_attn_varlen_func # type: ignore + from flash_attn.bert_padding import pad_input, unpad_input # type: ignore +except ImportError: + print("FlashAttention-2 is not installed, ignore this if you are not using FlashAttention.") + + +logger = logging.get_logger(__name__) + + +# Modified from: https://github.com/huggingface/transformers/blob/main/src/transformers/models/llama/modeling_llama.py +class LlamaShiftShortAttention(LlamaAttention): + + def forward( + self, + hidden_states: torch.Tensor, + attention_mask: Optional[torch.Tensor] = None, + position_ids: Optional[torch.LongTensor] = None, + past_key_value: Optional[Tuple[torch.Tensor]] = None, + output_attentions: bool = False, + use_cache: bool = False, + **kwargs + ) -> Tuple[torch.Tensor, Optional[torch.Tensor], Optional[Tuple[torch.Tensor]]]: + bsz, q_len, _ = hidden_states.size() + + query_states = self.q_proj(hidden_states) + key_states = self.k_proj(hidden_states) + value_states = self.v_proj(hidden_states) + + query_states = query_states.view(bsz, q_len, self.num_heads, self.head_dim).transpose(1, 2) + key_states = key_states.view(bsz, q_len, self.num_key_value_heads, self.head_dim).transpose(1, 2) + value_states = value_states.view(bsz, q_len, self.num_key_value_heads, self.head_dim).transpose(1, 2) + + kv_seq_len = key_states.shape[-2] + if past_key_value is not None: + kv_seq_len += past_key_value[0].shape[-2] + + cos, sin = self.rotary_emb(value_states, seq_len=kv_seq_len) + query_states, key_states = apply_rotary_pos_emb(query_states, key_states, cos, sin, position_ids) + + if past_key_value is not None: # reuse k, v, self_attention + key_states = torch.cat([past_key_value[0], key_states], dim=2) + value_states = torch.cat([past_key_value[1], value_states], dim=2) + + past_key_value = (key_states, value_states) if use_cache else None + + if getattr(self, "num_key_value_groups"): + key_states = repeat_kv(key_states, self.num_key_value_groups) + value_states = repeat_kv(value_states, self.num_key_value_groups) + + if getattr(self.config, "group_size_ratio", None) and self.training: # shift + groupsz = int(q_len * getattr(self.config, "group_size_ratio")) + assert q_len % groupsz == 0, "q_len {} should be divisible by group size {}.".format(q_len, groupsz) + num_groups = q_len // groupsz + def shift(state: torch.Tensor) -> torch.Tensor: + state = state.transpose(1, 2) # output: (bsz, seq_len, n_heads, head_dim) + state = torch.cat(( + state[:, :, :self.num_heads//2], state[:, :, self.num_heads//2:].roll(-groupsz//2, dims=1) + ), dim=2) + return state.reshape(bsz * num_groups, groupsz, self.num_heads, self.head_dim).transpose(1, 2) + + query_states, key_states, value_states = shift(query_states), shift(key_states), shift(value_states) + if attention_mask is not None: + attention_mask = attention_mask[:, :, :groupsz, :groupsz].repeat(num_groups, 1, 1, 1) + + attn_weights = torch.matmul(query_states, key_states.transpose(2, 3)) / math.sqrt(self.head_dim) + + if attention_mask is not None: + attn_weights = attn_weights + attention_mask + + # upcast attention to fp32 + attn_weights = nn.functional.softmax(attn_weights, dim=-1, dtype=torch.float32).to(query_states.dtype) + attn_output = torch.matmul(attn_weights, value_states) # (bsz, :, seq_len, :) or (bsz*n_group, :, groupsz, :) + attn_output = attn_output.transpose(1, 2).contiguous() + + if getattr(self.config, "group_size_ratio", None) and self.training: # shift back + attn_output.reshape(bsz, q_len, self.num_heads, self.head_dim) + attn_output = torch.cat(( + attn_output[:, :, :self.num_heads//2], attn_output[:, :, self.num_heads//2:].roll(groupsz//2, dims=1) + )) + + attn_output = attn_output.reshape(bsz, q_len, self.hidden_size) + attn_output = self.o_proj(attn_output) + + if not output_attentions: + attn_weights = None + + return attn_output, attn_weights, past_key_value + + +class LlamaFlashAttention2(LlamaAttention): + + def forward( + self, + hidden_states: torch.Tensor, + attention_mask: Optional[torch.Tensor] = None, + position_ids: Optional[torch.LongTensor] = None, + past_key_value: Optional[Tuple[torch.Tensor]] = None, + output_attentions: bool = False, + use_cache: bool = False, + **kwargs + ) -> Tuple[torch.Tensor, Optional[torch.Tensor], Optional[Tuple[torch.Tensor]]]: + # LlamaFlashAttention2 attention does not support output_attentions + output_attentions = False + + bsz, q_len, _ = hidden_states.size() + + query_states = self.q_proj(hidden_states) + key_states = self.k_proj(hidden_states) + value_states = self.v_proj(hidden_states) + + # FlashAttention requires the input to have the shape (bsz, seq_len, n_heads, head_dim) + query_states = query_states.view(bsz, q_len, self.num_heads, self.head_dim).transpose(1, 2) + key_states = key_states.view(bsz, q_len, self.num_key_value_heads, self.head_dim).transpose(1, 2) + value_states = value_states.view(bsz, q_len, self.num_key_value_heads, self.head_dim).transpose(1, 2) + + kv_seq_len = key_states.shape[-2] + if past_key_value is not None: + kv_seq_len += past_key_value[0].shape[-2] + + cos, sin = self.rotary_emb(value_states, seq_len=kv_seq_len) + query_states, key_states = apply_rotary_pos_emb(query_states, key_states, cos, sin, position_ids) + + if past_key_value is not None: # reuse k, v, self_attention + key_states = torch.cat([past_key_value[0], key_states], dim=2) + value_states = torch.cat([past_key_value[1], value_states], dim=2) + + past_key_value = (key_states, value_states) if use_cache else None + + # cast to half precision + input_dtype = query_states.dtype + if input_dtype == torch.float32: + logger.warning_once("The input hidden states seems to be silently casted in float32.") + query_states = query_states.to(self.config.torch_dtype) + key_states = key_states.to(self.config.torch_dtype) + value_states = value_states.to(self.config.torch_dtype) + + if getattr(self, "num_key_value_groups", None): + key_states = repeat_kv(key_states, self.num_key_value_groups) + value_states = repeat_kv(value_states, self.num_key_value_groups) + + query_states = query_states.transpose(1, 2) # (bsz, seq_len, n_heads, head_dim) + key_states = key_states.transpose(1, 2) # (bsz, seq_len, n_heads, head_dim) + value_states = value_states.transpose(1, 2) # (bsz, seq_len, n_heads, head_dim) + + if getattr(self.config, "group_size_ratio", None) and self.training: # shift + groupsz = int(q_len * getattr(self.config, "group_size_ratio")) + assert q_len % groupsz == 0, "q_len {} should be divisible by group size {}.".format(q_len, groupsz) + num_groups = q_len // groupsz + def shift(state: torch.Tensor) -> torch.Tensor: + state = torch.cat(( + state[:, :, :self.num_heads//2], state[:, :, self.num_heads//2:].roll(-groupsz//2, dims=1) + ), dim=2) + return state.reshape(bsz * num_groups, groupsz, self.num_heads, self.head_dim) + + query_states, key_states, value_states = shift(query_states), shift(key_states), shift(value_states) + if attention_mask is not None: + attention_mask = attention_mask.reshape(bsz * num_groups, groupsz) + + if attention_mask is not None: + logger.warning_once("Padded sequences are less efficient in FlashAttention.") + # -q_len: assumes left padding when q_len != kv_len + unpadded_q, indices_q, cu_seqlens_q, max_seqlen_q = unpad_input(query_states, attention_mask[:, -q_len:]) + unpadded_k, _, cu_seqlens_k, max_seqlen_k = unpad_input(key_states, attention_mask) + unpadded_v, _, _, _ = unpad_input(value_states, attention_mask) + attn_output_unpad = flash_attn_varlen_func( + unpadded_q, + unpadded_k, + unpadded_v, + cu_seqlens_q=cu_seqlens_q, + cu_seqlens_k=cu_seqlens_k, + max_seqlen_q=max_seqlen_q, + max_seqlen_k=max_seqlen_k, + dropout_p=0.0, + softmax_scale=None, + causal=True, + ) + attn_output = pad_input(attn_output_unpad, indices_q, bsz, q_len) + else: + attn_output = flash_attn_func( + query_states, key_states, value_states, 0.0, softmax_scale=None, causal=True + ) + + if getattr(self.config, "group_size_ratio", None) and self.training: # shift back + attn_output.reshape(bsz, q_len, self.num_heads, self.head_dim) + attn_output = torch.cat(( + attn_output[:, :, :self.num_heads//2], attn_output[:, :, self.num_heads//2:].roll(groupsz//2, dims=1) + )) + + attn_output = attn_output.reshape(bsz, q_len, self.hidden_size).contiguous() + attn_output = self.o_proj(attn_output) + + if not output_attentions: + attn_weights = None + + return attn_output, attn_weights, past_key_value + + +# Disable the transformation of the attention mask in LlamaModel as flash attention +# takes a boolean padding_mask. Fills in the past kv length for use in forward. +def _prepare_decoder_attention_mask( + self, + attention_mask: torch.Tensor, + input_shape: torch.Tensor, + inputs_embeds: torch.Tensor, + past_key_values_length: int +) -> torch.Tensor: + if attention_mask is not None and torch.all(attention_mask): + return None # This uses the faster call when training with full samples + + return attention_mask diff --git a/llm_rl/src/llmtuner/extras/ploting.py b/llm_rl/src/llmtuner/extras/ploting.py new file mode 100644 index 00000000..82530e45 --- /dev/null +++ b/llm_rl/src/llmtuner/extras/ploting.py @@ -0,0 +1,52 @@ +import os +import math +import json +import matplotlib.pyplot as plt +from typing import List, Optional +from transformers.trainer import TRAINER_STATE_NAME + +from llmtuner.extras.logging import get_logger + + +logger = get_logger(__name__) + + +def smooth(scalars: List[float]) -> List[float]: + r""" + EMA implementation according to TensorBoard. + """ + last = scalars[0] + smoothed = list() + weight = 1.8 * (1 / (1 + math.exp(-0.05 * len(scalars))) - 0.5) # a sigmoid function + for next_val in scalars: + smoothed_val = last * weight + (1 - weight) * next_val + smoothed.append(smoothed_val) + last = smoothed_val + return smoothed + + +def plot_loss(save_dictionary: os.PathLike, keys: Optional[List[str]] = ["loss"]) -> None: + + with open(os.path.join(save_dictionary, TRAINER_STATE_NAME), "r", encoding="utf-8") as f: + data = json.load(f) + + for key in keys: + steps, metrics = [], [] + for i in range(len(data["log_history"])): + if key in data["log_history"][i]: + steps.append(data["log_history"][i]["step"]) + metrics.append(data["log_history"][i][key]) + + if len(metrics) == 0: + logger.warning(f"No metric {key} to plot.") + continue + + plt.figure() + plt.plot(steps, metrics, alpha=0.4, label="original") + plt.plot(steps, smooth(metrics), label="smoothed") + plt.title("training {} of {}".format(key, save_dictionary)) + plt.xlabel("step") + plt.ylabel(key) + plt.legend() + plt.savefig(os.path.join(save_dictionary, "training_{}.png".format(key)), format="png", dpi=100) + print("Figure saved:", os.path.join(save_dictionary, "training_{}.png".format(key))) diff --git a/llm_rl/src/llmtuner/extras/save_and_load.py b/llm_rl/src/llmtuner/extras/save_and_load.py new file mode 100644 index 00000000..6d819ce6 --- /dev/null +++ b/llm_rl/src/llmtuner/extras/save_and_load.py @@ -0,0 +1,21 @@ +import os +import torch +from transformers.trainer import WEIGHTS_NAME + +from llmtuner.extras.logging import get_logger + + +logger = get_logger(__name__) + + +def load_valuehead_params(model: torch.nn.Module, checkpoint_dir: os.PathLike) -> bool: + vhead_file = os.path.join(checkpoint_dir, WEIGHTS_NAME) + if not os.path.exists(vhead_file): + logger.warning("Provided path ({}) does not contain valuehead weights.".format(checkpoint_dir)) + return False + vhead_params = torch.load(vhead_file, map_location="cpu") + model.register_buffer("reward_head_weight", vhead_params["v_head.summary.weight"], persistent=False) + model.register_buffer("reward_head_bias", vhead_params["v_head.summary.bias"], persistent=False) + model.register_buffer("default_head_weight", torch.zeros_like(vhead_params["v_head.summary.weight"]), persistent=False) + model.register_buffer("default_head_bias", torch.zeros_like(vhead_params["v_head.summary.bias"]), persistent=False) + return True diff --git a/llm_rl/src/llmtuner/extras/template.py b/llm_rl/src/llmtuner/extras/template.py new file mode 100644 index 00000000..401750ce --- /dev/null +++ b/llm_rl/src/llmtuner/extras/template.py @@ -0,0 +1,713 @@ +import tiktoken +from dataclasses import dataclass +from typing import TYPE_CHECKING, Dict, List, Optional, Tuple, Union + +from llmtuner.extras.logging import get_logger + +if TYPE_CHECKING: + from transformers import PreTrainedTokenizer + + +logger = get_logger(__name__) + + +@dataclass +class Template: + + prefix: List[Union[str, Dict[str, str]]] + prompt: List[Union[str, Dict[str, str]]] + system: str + sep: List[Union[str, Dict[str, str]]] + stop_words: List[str] + use_history: bool + efficient_eos: bool + + def encode_oneturn( + self, + tokenizer: "PreTrainedTokenizer", + query: str, + resp: str, + history: Optional[List[Tuple[str, str]]] = None, + system: Optional[str] = None + ) -> Tuple[List[int], List[int]]: + r""" + Returns a single pair of token ids representing prompt and response respectively. + """ + system, history = self._format(query, resp, history, system) + encoded_pairs = self._encode(tokenizer, system, history) + prompt_ids = [] + for query_ids, resp_ids in encoded_pairs[:-1]: + prompt_ids = prompt_ids + query_ids + resp_ids + prompt_ids, answer_ids = prompt_ids + encoded_pairs[-1][0], encoded_pairs[-1][1] + return prompt_ids, answer_ids + + def encode_multiturn( + self, + tokenizer: "PreTrainedTokenizer", + query: str, + resp: str, + history: Optional[List[Tuple[str, str]]] = None, + system: Optional[str] = None + ) -> List[Tuple[List[int], List[int]]]: + r""" + Returns multiple pairs of token ids representing prompts and responses respectively. + """ + system, history = self._format(query, resp, history, system) + encoded_pairs = self._encode(tokenizer, system, history) + return encoded_pairs + + def _format( + self, + query: str, + resp: str, + history: Optional[List[Tuple[str, str]]] = None, + system: Optional[str] = None + ) -> Tuple[str, List[Tuple[str, str]]]: + r""" + Aligns inputs to the standard format. + """ + system = system or self.system # use system if provided + history = history if (history and self.use_history) else [] + history = history + [(query, resp)] + return system, history + + def _get_special_ids( + self, + tokenizer: "PreTrainedTokenizer" + ) -> Tuple[List[int], List[int]]: + if tokenizer.bos_token_id is not None and getattr(tokenizer, "add_bos_token", True): + bos_ids = [tokenizer.bos_token_id] + else: # baichuan, qwen and gpt2 models have no bos token + bos_ids = [] + + if tokenizer.eos_token_id is None: + raise ValueError("EOS token is required.") + + if self.efficient_eos: # used in baichuan, qwen, chatglm, etc. + eos_ids = [] + else: + eos_ids = [tokenizer.eos_token_id] + + return bos_ids, eos_ids + + def _encode( + self, + tokenizer: "PreTrainedTokenizer", + system: str, + history: List[Tuple[str, str]] + ) -> List[Tuple[List[int], List[int]]]: + r""" + Encodes formatted inputs to pairs of token ids. + Turn 0: bos + prefix + sep + query resp + eos + Turn t: sep + bos + query resp + eos + """ + bos_ids, eos_ids = self._get_special_ids(tokenizer) + sep_ids = self._convert_inputs_to_ids(tokenizer, context=self.sep) + encoded_pairs = [] + for turn_idx, (query, resp) in enumerate(history): + if turn_idx == 0: + prefix_ids = self._convert_inputs_to_ids(tokenizer, context=self.prefix, system=system) + if len(prefix_ids) != 0: # has prefix + prefix_ids = bos_ids + prefix_ids + sep_ids + else: + prefix_ids = bos_ids + else: + prefix_ids = sep_ids + bos_ids + + query_ids = self._convert_inputs_to_ids(tokenizer, context=self.prompt, query=query, idx=str(turn_idx)) + resp_ids = self._convert_inputs_to_ids(tokenizer, context=[resp]) + encoded_pairs.append((prefix_ids + query_ids, resp_ids + eos_ids)) + return encoded_pairs + + def _convert_inputs_to_ids( + self, + tokenizer: "PreTrainedTokenizer", + context: List[Union[str, Dict[str, str]]], + system: Optional[str] = None, + query: Optional[str] = None, + idx: Optional[str] = None + ) -> List[int]: + r""" + Converts context to token ids. + """ + if isinstance(getattr(tokenizer, "tokenizer", None), tiktoken.Encoding): # for tiktoken tokenizer (Qwen) + kwargs = dict(allowed_special="all") + else: + kwargs = dict(add_special_tokens=False) + + token_ids = [] + for elem in context: + if isinstance(elem, str): + elem = elem.replace("{{system}}", system, 1) if system is not None else elem + elem = elem.replace("{{query}}", query, 1) if query is not None else elem + elem = elem.replace("{{idx}}", idx, 1) if idx is not None else elem + if len(elem) != 0: + token_ids = token_ids + tokenizer.encode(elem, **kwargs) + elif isinstance(elem, dict): + token_ids = token_ids + [tokenizer.convert_tokens_to_ids(elem.get("token"))] + else: + raise ValueError("Input must be string or dict[str, str], got {}".format(type(elem))) + + return token_ids + + +@dataclass +class Llama2Template(Template): + + def _encode( + self, + tokenizer: "PreTrainedTokenizer", + system: str, + history: List[Tuple[str, str]] + ) -> List[Tuple[List[int], List[int]]]: + r""" + Encodes formatted inputs to pairs of token ids. + Turn 0: bos + prefix + query resp + eos + Turn t: bos + query resp + eos + """ + bos_ids, eos_ids = self._get_special_ids(tokenizer) + encoded_pairs = [] + for turn_idx, (query, resp) in enumerate(history): + if turn_idx == 0: # llama2 template has no sep_ids + query = self.prefix[0].replace("{{system}}", system) + query + query_ids = self._convert_inputs_to_ids(tokenizer, context=self.prompt, query=query) + resp_ids = self._convert_inputs_to_ids(tokenizer, context=[resp]) + encoded_pairs.append((bos_ids + query_ids, resp_ids + eos_ids)) + return encoded_pairs + + +templates: Dict[str, Template] = {} + + +def register_template( + name: str, + prefix: List[Union[str, Dict[str, str]]], + prompt: List[Union[str, Dict[str, str]]], + system: str, + sep: List[Union[str, Dict[str, str]]], + stop_words: Optional[List[str]] = [], + use_history: Optional[bool] = True, + efficient_eos: Optional[bool] = False +) -> None: + template_class = Llama2Template if "llama2" in name else Template + templates[name] = template_class( + prefix=prefix, + prompt=prompt, + system=system, + sep=sep, + stop_words=stop_words, + use_history=use_history, + efficient_eos=efficient_eos + ) + + +def get_template_and_fix_tokenizer( + name: str, + tokenizer: "PreTrainedTokenizer" +) -> Template: + if tokenizer.eos_token_id is None: + tokenizer.eos_token = "<|endoftext|>" + logger.info("Add eos token: {}".format(tokenizer.eos_token)) + + if tokenizer.pad_token_id is None: + tokenizer.pad_token = tokenizer.eos_token + logger.info("Add pad token: {}".format(tokenizer.pad_token)) + + if name is None: + return None + + template = templates.get(name, None) + assert template is not None, "Template {} does not exist.".format(name) + tokenizer.add_special_tokens( + dict(additional_special_tokens=template.stop_words), + replace_additional_special_tokens=False + ) + return template + + +r""" +Supports: https://huggingface.co/tatsu-lab/alpaca-7b-wdiff +""" +register_template( + name="alpaca", + prefix=[ + "{{system}}" + ], + prompt=[ + "### Instruction:\n{{query}}\n\n### Response:\n" + ], + system=( + "Below is an instruction that describes a task. " + "Write a response that appropriately completes the request." + ), + sep=[ + "\n\n" + ] +) + + +r""" +Supports: https://huggingface.co/BAAI/AquilaChat-7B + https://huggingface.co/BAAI/AquilaChat2-7B + https://huggingface.co/BAAI/AquilaChat2-34B +""" +register_template( + name="aquila", + prefix=[ + "{{system}}" + ], + prompt=[ + "Human: {{query}}###Assistant:" + ], + system=( + "A chat between a curious human and an artificial intelligence assistant. " + "The assistant gives helpful, detailed, and polite answers to the human's questions." + ), + sep=[ + "###" + ], + stop_words=[ + "" + ], + efficient_eos=True +) + + +r""" +Supports: https://huggingface.co/baichuan-inc/Baichuan-13B-Chat +""" +register_template( + name="baichuan", + prefix=[ + "{{system}}" + ], + prompt=[ + {"token": ""}, # user token + "{{query}}", + {"token": ""} # assistant token + ], + system="", + sep=[], + efficient_eos=True +) + + +r""" +Supports: https://huggingface.co/baichuan-inc/Baichuan2-7B-Chat + https://huggingface.co/baichuan-inc/Baichuan2-13B-Chat +""" +register_template( + name="baichuan2", + prefix=[ + "{{system}}" + ], + prompt=[ + {"token": ""}, # user token + "{{query}}", + {"token": ""} # assistant token + ], + system="", + sep=[], + efficient_eos=True +) + + +r""" +Supports: https://huggingface.co/BelleGroup/BELLE-LLaMA-EXT-13B +""" +register_template( + name="belle", + prefix=[ + "{{system}}" + ], + prompt=[ + "Human: {{query}}\n\nBelle: " + ], + system="", + sep=[ + "\n\n" + ] +) + + +r""" +Supports: https://huggingface.co/vivo-ai/BlueLM-7B-Chat +""" +register_template( + name="bluelm", + prefix=[ + "{{system}}" + ], + prompt=[ + {"token": "[|Human|]:"}, + "{{query}}", + {"token": "[|AI|]:"} + ], + system="", + sep=[] +) + + +r""" +Supports: https://huggingface.co/THUDM/chatglm2-6b +""" +register_template( + name="chatglm2", + prefix=[ + {"token": "[gMASK]"}, + {"token": "sop"}, + "{{system}}" + ], + prompt=[ + "[Round {{idx}}]\n\n问:{{query}}\n\n答:" + ], + system="", + sep=[ + "\n\n" + ], + efficient_eos=True +) + + +r""" +Supports: https://huggingface.co/THUDM/chatglm3-6b +""" +register_template( + name="chatglm3", + prefix=[ + {"token": "[gMASK]"}, + {"token": "sop"}, + "{{system}}" + ], + prompt=[ + {"token": "<|user|>"}, + "\n", + "{{query}}", + {"token": "<|assistant|>"} + ], + system="", + sep=[], + stop_words=[ + "<|user|>", + "<|observation|>" + ], + efficient_eos=True +) + + +r""" +Supports: https://huggingface.co/deepseek-ai/deepseek-coder-1.3b-instruct + https://huggingface.co/deepseek-ai/deepseek-coder-6.7b-instruct + https://huggingface.co/deepseek-ai/deepseek-coder-33b-instruct +""" +register_template( + name="deepseek", + prefix=[ + "{{system}}" + ], + prompt=[ + "### Instruction:\n{{query}}\n\n### Response:\n" + ], + system=( + "You are an AI programming assistant, utilizing the Deepseek Coder model, " + "developed by Deepseek Company, and you only answer questions related to computer science. " + "For politically sensitive questions, security and privacy issues, " + "and other non-computer science questions, you will refuse to answer." + ), + sep=[ + "\n", + {"token": "<|EOT|>"}, + "\n\n" + ], + stop_words=[ + "<|EOT|>" + ], + efficient_eos=True +) + + +r""" +Default template. +""" +register_template( + name="default", + prefix=[ + "{{system}}" + ], + prompt=[ + "Human: {{query}}\nAssistant:" + ], + system=( + "A chat between a curious user and an artificial intelligence assistant. " + "The assistant gives helpful, detailed, and polite answers to the user's questions." + ), + sep=[ + "\n" + ] +) + + +r""" +Supports: https://huggingface.co/internlm/internlm-chat-7b + https://huggingface.co/internlm/internlm-chat-20b +""" +register_template( + name="intern", + prefix=[ + "{{system}}" + ], + prompt=[ + "<|User|>:{{query}}", + {"token": ""}, + "\n<|Bot|>:" + ], + system="", + sep=[ + {"token": ""}, + "\n" + ], + stop_words=[ + "" + ], + efficient_eos=True +) + + +r""" +Supports: https://huggingface.co/meta-llama/Llama-2-7b-chat-hf + https://huggingface.co/meta-llama/Llama-2-13b-chat-hf + https://huggingface.co/meta-llama/Llama-2-70b-chat-hf +""" +register_template( + name="llama2", + prefix=[ + "<>\n{{system}}\n<>\n\n" + ], + prompt=[ + "[INST] {{query}} [/INST]" + ], + system=( + "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.\n\n" + "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." + ), + sep=[] +) + + +r""" +Supports: https://huggingface.co/ziqingyang/chinese-alpaca-2-7b + https://huggingface.co/ziqingyang/chinese-alpaca-2-13b +""" +register_template( + name="llama2_zh", + prefix=[ + "<>\n{{system}}\n<>\n\n" + ], + prompt=[ + "[INST] {{query}} [/INST]" + ], + system="You are a helpful assistant. 你是一个乐于助人的助手。", + sep=[] +) + + +r""" +Supports: https://huggingface.co/mistralai/Mistral-7B-Instruct-v0.1 +""" +register_template( + name="mistral", + prefix=[ + "{{system}}" + ], + prompt=[ + "[INST] {{query}} [/INST]" + ], + system="", + sep=[] +) + + +r""" +Supports: https://huggingface.co/openchat/openchat_3.5 +""" +register_template( + name="openchat", + prefix=[ + "{{system}}" + ], + prompt=[ + "GPT4 Correct User: {{query}}", + {"token": "<|end_of_turn|>"}, + "GPT4 Correct Assistant:" + ], + system="You are a helpful assistant.", + sep=[ + {"token": "<|end_of_turn|>"} + ], + stop_words=[ + "<|end_of_turn|>" + ], + efficient_eos=True +) + + +r""" +Supports: https://huggingface.co/Qwen/Qwen-7B-Chat + https://huggingface.co/Qwen/Qwen-14B-Chat +""" +register_template( + name="qwen", + prefix=[ + {"token": "<|im_start|>"}, + "system\n{{system}}" + ], + prompt=[ + {"token": "<|im_start|>"}, + "user\n{{query}}", + {"token": "<|im_end|>"}, + "\n", + {"token": "<|im_start|>"}, + "assistant\n" + ], + system="You are a helpful assistant.", + sep=[ + {"token": "<|im_end|>"}, + "\n" + ], + stop_words=[ + "<|im_end|>" + ], + efficient_eos=True +) + + +r""" +Supports: https://huggingface.co/HuggingFaceH4/starchat-alpha + https://huggingface.co/HuggingFaceH4/starchat-beta +""" +register_template( + name="starchat", + prefix=[ + {"token": "<|system|>"}, + "\n{{system}}", + ], + prompt=[ + {"token": "<|user|>"}, + "\n{{query}}", + {"token": "<|end|>"}, + "\n", + {"token": "<|assistant|>"} + ], + system="", + sep=[ + {"token": "<|end|>"}, + "\n" + ], + stop_words=[ + "<|end|>" + ], + efficient_eos=True +) + + +r""" +Supports language model inference without histories. +""" +register_template( + name="vanilla", + prefix=[], + prompt=[ + "{{query}}" + ], + system="", + sep=[], + use_history=False +) + + +r""" +Supports: https://huggingface.co/lmsys/vicuna-7b-v1.5 + https://huggingface.co/lmsys/vicuna-13b-v1.5 +""" +register_template( + name="vicuna", + prefix=[ + "{{system}}" + ], + prompt=[ + "USER: {{query}} ASSISTANT:" + ], + system=( + "A chat between a curious user and an artificial intelligence assistant. " + "The assistant gives helpful, detailed, and polite answers to the user's questions." + ), + sep=[] +) + + +r""" +Supports: https://huggingface.co/xverse/XVERSE-7B-Chat + https://huggingface.co/xverse/XVERSE-13B-Chat +""" +register_template( + name="xverse", + prefix=[ + "{{system}}" + ], + prompt=[ + "Human: {{query}}\n\nAssistant: " + ], + system="", + sep=[] +) + + +r""" +Supports: https://huggingface.co/HuggingFaceH4/zephyr-7b-alpha + https://huggingface.co/HuggingFaceH4/zephyr-7b-beta +""" +register_template( + name="zephyr", + prefix=[ + {"token": "<|system|>"}, + "\n{{system}}", + {"token": ""} + ], + prompt=[ + {"token": "<|user|>"}, + "\n{{query}}", + {"token": ""}, + {"token": "<|assistant|>"} + ], + system="You are a friendly chatbot who always responds in the style of a pirate", + sep=[] +) + + +r""" +Supports: https://huggingface.co/IDEA-CCNL/Ziya-LLaMA-13B-v1 + https://huggingface.co/IDEA-CCNL/Ziya-LLaMA-13B-v1.1 + https://huggingface.co/IDEA-CCNL/Ziya2-13B-Chat +""" +register_template( + name="ziya", + prefix=[ + "{{system}}" + ], + prompt=[ + {"token": ""}, + ":{{query}}\n", + {"token": ""}, + ":" + ], + system="", + sep=[ + "\n" + ] +) diff --git a/llm_rl/src/llmtuner/hparams/__init__.py b/llm_rl/src/llmtuner/hparams/__init__.py new file mode 100644 index 00000000..f0547cc5 --- /dev/null +++ b/llm_rl/src/llmtuner/hparams/__init__.py @@ -0,0 +1,4 @@ +from .data_args import DataArguments +from .finetuning_args import FinetuningArguments +from .generating_args import GeneratingArguments +from .model_args import ModelArguments diff --git a/llm_rl/src/llmtuner/hparams/data_args.py b/llm_rl/src/llmtuner/hparams/data_args.py new file mode 100644 index 00000000..4c67dd65 --- /dev/null +++ b/llm_rl/src/llmtuner/hparams/data_args.py @@ -0,0 +1,169 @@ +import os +import json +from typing import List, Literal, Optional +from dataclasses import dataclass, field + + +@dataclass +class DatasetAttr: + + load_from: str + dataset_name: Optional[str] = None + dataset_sha1: Optional[str] = None + system_prompt: Optional[str] = None + subset: Optional[str] = None + ranking: Optional[bool] = False + formatting: Optional[Literal["alpaca", "sharegpt"]] = "alpaca" + + prompt: Optional[str] = "instruction" + query: Optional[str] = "input" + response: Optional[str] = "output" + history: Optional[str] = None + messages: Optional[str] = "conversations" + role: Optional[str] = "from" + content: Optional[str] = "value" + + def __repr__(self) -> str: + return self.dataset_name + + +@dataclass +class DataArguments: + r""" + Arguments pertaining to what data we are going to input our model for training and evaluation. + """ + template: Optional[str] = field( + default=None, + metadata={"help": "Which template to use for constructing prompts in training and inference."} + ) + dataset: Optional[str] = field( + default=None, + metadata={"help": "The name of provided dataset(s) to use. Use commas to separate multiple datasets."} + ) + dataset_dir: Optional[str] = field( + default="data", + metadata={"help": "The name of the folder containing datasets."} + ) + split: Optional[str] = field( + default="train", + metadata={"help": "Which dataset split to use for training and evaluation."} + ) + cutoff_len: Optional[int] = field( + default=1024, + metadata={"help": "The maximum length of the model inputs after tokenization."} + ) + train_on_prompt: Optional[bool] = field( + default=False, + metadata={"help": "Whether to disable the mask on the prompt or not."} + ) + streaming: Optional[bool] = field( + default=False, + metadata={"help": "Enable dataset streaming."} + ) + buffer_size: Optional[int] = field( + default=16384, + metadata={"help": "Size of the buffer to randomly sample examples from in dataset streaming."} + ) + mix_strategy: Optional[Literal["concat", "interleave_under", "interleave_over"]] = field( + default="concat", + metadata={"help": "Strategy to use in dataset mixing (concat/interleave) (undersampling/oversampling)."} + ) + interleave_probs: Optional[str] = field( + default=None, + metadata={"help": "Probabilities to sample data from datasets. Use commas to separate multiple datasets."} + ) + overwrite_cache: Optional[bool] = field( + default=False, + metadata={"help": "Overwrite the cached training and evaluation sets."} + ) + preprocessing_num_workers: Optional[int] = field( + default=None, + metadata={"help": "The number of processes to use for the preprocessing."} + ) + max_samples: Optional[int] = field( + default=None, + metadata={"help": "For debugging purposes, truncate the number of examples for each dataset."} + ) + eval_num_beams: Optional[int] = field( + default=None, + metadata={"help": "Number of beams to use for evaluation. This argument will be passed to `model.generate`"} + ) + ignore_pad_token_for_loss: Optional[bool] = field( + default=True, + metadata={"help": "Whether to ignore the tokens corresponding to padded labels in the loss computation or not."} + ) + system_prompt: Optional[str] = field( + default=None, + metadata={"help": "System prompt to add before the user query. Use `|` to separate multiple prompts in training."} + ) + val_size: Optional[float] = field( + default=0, + metadata={"help": "Size of the development set, should be an integer or a float in range `[0,1)`."} + ) + sft_packing: Optional[bool] = field( + default=False, + metadata={"help": "Packing the questions and answers in the supervised fine-tuning stage."} + ) + cache_path: Optional[str] = field( + default=None, + metadata={"help": "Path to save or load the preprocessed datasets."} + ) + + def __post_init__(self): + if self.streaming and self.val_size > 1e-6 and self.val_size < 1: + raise ValueError("Streaming mode should have an integer val size.") + + if self.streaming and self.max_samples is not None: + raise ValueError("`max_samples` is incompatible with `streaming`.") + + if self.streaming and self.cache_path: + raise ValueError("`cache_path` is incompatible with `streaming`.") + + def init_for_training(self, seed: int): # support mixing multiple datasets + self.seed = seed + dataset_names = [ds.strip() for ds in self.dataset.split(",")] if self.dataset is not None else [] + try: + with open(os.path.join(self.dataset_dir, "dataset_info.json"), "r") as f: + dataset_info = json.load(f) + except Exception: + if self.dataset is not None: + raise ValueError("Cannot find dataset_info.json in `dataset_dir`.") + dataset_info = None + + prompt_list = self.system_prompt.split("|") if self.system_prompt else [None] + prompt_list = prompt_list * (len(dataset_names) // len(prompt_list)) + assert len(prompt_list) == len(dataset_names), "Number of system prompts should be equal to datasets or 1." + + if self.interleave_probs is not None: + self.interleave_probs = [float(prob.strip()) for prob in self.interleave_probs.split(",")] + + self.dataset_list: List[DatasetAttr] = [] + for i, name in enumerate(dataset_names): + if name not in dataset_info: + raise ValueError("Undefined dataset {} in dataset_info.json.".format(name)) + + if "hf_hub_url" in dataset_info[name]: + dataset_attr = DatasetAttr("hf_hub", dataset_name=dataset_info[name]["hf_hub_url"]) + elif "script_url" in dataset_info[name]: + dataset_attr = DatasetAttr("script", dataset_name=dataset_info[name]["script_url"]) + else: + dataset_attr = DatasetAttr( + "file", + dataset_name=dataset_info[name]["file_name"], + dataset_sha1=dataset_info[name].get("file_sha1", None) + ) + + if "columns" in dataset_info[name]: + dataset_attr.prompt = dataset_info[name]["columns"].get("prompt", None) + dataset_attr.query = dataset_info[name]["columns"].get("query", None) + dataset_attr.response = dataset_info[name]["columns"].get("response", None) + dataset_attr.history = dataset_info[name]["columns"].get("history", None) + dataset_attr.messages = dataset_info[name]["columns"].get("messages", None) + dataset_attr.role = dataset_info[name]["columns"].get("role", None) + dataset_attr.content = dataset_info[name]["columns"].get("content", None) + + dataset_attr.subset = dataset_info[name].get("subset", None) + dataset_attr.ranking = dataset_info[name].get("ranking", False) + dataset_attr.formatting = dataset_info[name].get("formatting", "alpaca") + dataset_attr.system_prompt = prompt_list[i] + self.dataset_list.append(dataset_attr) diff --git a/llm_rl/src/llmtuner/hparams/finetuning_args.py b/llm_rl/src/llmtuner/hparams/finetuning_args.py new file mode 100644 index 00000000..d5ef323d --- /dev/null +++ b/llm_rl/src/llmtuner/hparams/finetuning_args.py @@ -0,0 +1,107 @@ +import json +from typing import Literal, Optional +from dataclasses import asdict, dataclass, field + + +@dataclass +class FinetuningArguments: + r""" + Arguments pertaining to which techniques we are going to fine-tuning with. + """ + stage: Optional[Literal["pt", "sft", "rm", "ppo", "dpo"]] = field( + default="sft", + metadata={"help": "Which stage will be performed in training."} + ) + finetuning_type: Optional[Literal["lora", "freeze", "full", "none"]] = field( + default="lora", + metadata={"help": "Which fine-tuning method to use."} + ) + num_layer_trainable: Optional[int] = field( + default=3, + metadata={"help": "Number of trainable layers for partial-parameter (freeze) fine-tuning."} + ) + name_module_trainable: Optional[Literal["mlp", "self_attn", "self_attention"]] = field( + default="mlp", + metadata={"help": "Name of trainable modules for partial-parameter (freeze) fine-tuning. \ + LLaMA choices: [\"mlp\", \"self_attn\"], \ + BLOOM & Falcon & ChatGLM2 choices: [\"mlp\", \"self_attention\"], \ + Qwen choices: [\"mlp\", \"attn\"], \ + Phi-1.5 choices: [\"mlp\", \"mixer\"], \ + LLaMA-2, Baichuan, InternLM, XVERSE choices: the same as LLaMA."} + ) + lora_rank: Optional[int] = field( + default=8, + metadata={"help": "The intrinsic dimension for LoRA fine-tuning."} + ) + lora_alpha: Optional[float] = field( + default=32.0, + metadata={"help": "The scale factor for LoRA fine-tuning (similar with the learning rate)."} + ) + lora_dropout: Optional[float] = field( + default=0.1, + metadata={"help": "Dropout rate for the LoRA fine-tuning."} + ) + lora_target: Optional[str] = field( + default=None, + metadata={"help": "Name(s) of target modules to apply LoRA. Use commas to separate multiple modules. \ + LLaMA choices: [\"q_proj\", \"k_proj\", \"v_proj\", \"o_proj\", \"gate_proj\", \"up_proj\", \"down_proj\"], \ + BLOOM & Falcon & ChatGLM2 choices: [\"query_key_value\", \"self_attention.dense\", \"mlp.dense\"], \ + Baichuan choices: [\"W_pack\", \"o_proj\", \"gate_proj\", \"up_proj\", \"down_proj\"], \ + Qwen choices: [\"c_attn\", \"attn.c_proj\", \"w1\", \"w2\", \"mlp.c_proj\"], \ + Phi-1.5 choices: [\"Wqkv\", \"out_proj\", \"fc1\", \"fc2\"], \ + LLaMA-2, InternLM, XVERSE choices: the same as LLaMA."} + ) + additional_target: Optional[str] = field( + default=None, + metadata={"help": "Name(s) of modules apart from LoRA layers to be set as trainable and saved in the final checkpoint."} + ) + resume_lora_training: Optional[bool] = field( + default=True, + metadata={"help": "Whether to resume training from the last LoRA weights or create new weights after merging them."} + ) + ppo_score_norm: Optional[bool] = field( + default=False, + metadata={"help": "Use score normalization in PPO training."} + ) + ppo_logger: Optional[str] = field( + default=None, + metadata={"help": "Log with either 'wandb' or 'tensorboard' in PPO training."} + ) + ppo_target: Optional[float] = field( + default=6.0, + metadata={"help": "Target KL value for adaptive KL control in PPO training."} + ) + dpo_beta: Optional[float] = field( + default=0.1, + metadata={"help": "The beta parameter for the DPO loss."} + ) + upcast_layernorm: Optional[bool] = field( + default=False, + metadata={"help": "Whether to upcast the layernorm weights in fp32."} + ) + neft_alpha: Optional[float] = field( + default=0, + metadata={"help": "The alpha parameter to control the noise magnitude in NEFTune."} + ) + + def __post_init__(self): + if isinstance(self.lora_target, str): # support custom target modules/layers of LoRA + self.lora_target = [target.strip() for target in self.lora_target.split(",")] + + if isinstance(self.additional_target, str): + self.additional_target = [target.strip() for target in self.additional_target.split(",")] + + assert self.finetuning_type in ["lora", "freeze", "full", "none"], "Invalid fine-tuning method." + + def save_to_json(self, json_path: str): + r"""Saves the content of this instance in JSON format inside `json_path`.""" + json_string = json.dumps(asdict(self), indent=2, sort_keys=True) + "\n" + with open(json_path, "w", encoding="utf-8") as f: + f.write(json_string) + + @classmethod + def load_from_json(cls, json_path: str): + r"""Creates an instance from the content of `json_path`.""" + with open(json_path, "r", encoding="utf-8") as f: + text = f.read() + return cls(**json.loads(text)) diff --git a/llm_rl/src/llmtuner/hparams/general_args.py b/llm_rl/src/llmtuner/hparams/general_args.py new file mode 100644 index 00000000..c0c1a0de --- /dev/null +++ b/llm_rl/src/llmtuner/hparams/general_args.py @@ -0,0 +1,13 @@ +from typing import Literal, Optional +from dataclasses import dataclass, field + + +@dataclass +class GeneralArguments: + r""" + Arguments pertaining to which stage we are going to perform. + """ + stage: Optional[Literal["pt", "sft", "rm", "ppo", "dpo"]] = field( + default="sft", + metadata={"help": "Which stage will be performed in training."} + ) diff --git a/llm_rl/src/llmtuner/hparams/generating_args.py b/llm_rl/src/llmtuner/hparams/generating_args.py new file mode 100644 index 00000000..c04a5c36 --- /dev/null +++ b/llm_rl/src/llmtuner/hparams/generating_args.py @@ -0,0 +1,53 @@ +from typing import Any, Dict, Optional +from dataclasses import asdict, dataclass, field + + +@dataclass +class GeneratingArguments: + r""" + Arguments pertaining to specify the decoding parameters. + """ + do_sample: Optional[bool] = field( + default=True, + metadata={"help": "Whether or not to use sampling, use greedy decoding otherwise."} + ) + temperature: Optional[float] = field( + default=0.95, + metadata={"help": "The value used to modulate the next token probabilities."} + ) + top_p: Optional[float] = field( + default=0.7, + metadata={"help": "The smallest set of most probable tokens with probabilities that add up to top_p or higher are kept."} + ) + top_k: Optional[int] = field( + default=50, + metadata={"help": "The number of highest probability vocabulary tokens to keep for top-k filtering."} + ) + num_beams: Optional[int] = field( + default=1, + metadata={"help": "Number of beams for beam search. 1 means no beam search."} + ) + max_length: Optional[int] = field( + default=512, + metadata={"help": "The maximum length the generated tokens can have. It can be overridden by max_new_tokens."} + ) + max_new_tokens: Optional[int] = field( + default=512, + metadata={"help": "The maximum numbers of tokens to generate, ignoring the number of tokens in the prompt."} + ) + repetition_penalty: Optional[float] = field( + default=1.0, + metadata={"help": "The parameter for repetition penalty. 1.0 means no penalty."} + ) + length_penalty: Optional[float] = field( + default=1.0, + metadata={"help": "Exponential penalty to the length that is used with beam-based generation."} + ) + + def to_dict(self) -> Dict[str, Any]: + args = asdict(self) + if args.get("max_new_tokens", -1) > 0: + args.pop("max_length", None) + else: + args.pop("max_new_tokens", None) + return args diff --git a/llm_rl/src/llmtuner/hparams/model_args.py b/llm_rl/src/llmtuner/hparams/model_args.py new file mode 100644 index 00000000..7c25fad1 --- /dev/null +++ b/llm_rl/src/llmtuner/hparams/model_args.py @@ -0,0 +1,93 @@ +from typing import Literal, Optional +from dataclasses import dataclass, field + + +@dataclass +class ModelArguments: + r""" + Arguments pertaining to which model/config/tokenizer we are going to fine-tune. + """ + model_name_or_path: str = field( + metadata={"help": "Path to pretrained model or model identifier from huggingface.co/models."} + ) + cache_dir: Optional[str] = field( + default=None, + metadata={"help": "Where to store the pretrained models downloaded from huggingface.co."} + ) + use_fast_tokenizer: Optional[bool] = field( + default=True, + metadata={"help": "Whether to use one of the fast tokenizer (backed by the tokenizers library) or not."} + ) + split_special_tokens: Optional[bool] = field( + default=False, + metadata={"help": "Whether or not the special tokens should be split during the tokenization process."} + ) + use_auth_token: Optional[bool] = field( + default=False, + metadata={"help": "Will use the token generated when running `huggingface-cli login`."} + ) + model_revision: Optional[str] = field( + default="main", + metadata={"help": "The specific model version to use (can be a branch name, tag name or commit id)."} + ) + quantization_bit: Optional[int] = field( + default=None, + metadata={"help": "The number of bits to quantize the model."} + ) + quantization_type: Optional[Literal["fp4", "nf4"]] = field( + default="nf4", + metadata={"help": "Quantization data type to use in int4 training."} + ) + double_quantization: Optional[bool] = field( + default=True, + metadata={"help": "Whether to use double quantization in int4 training or not."} + ) + rope_scaling: Optional[Literal["linear", "dynamic"]] = field( + default=None, + metadata={"help": "Adopt scaled rotary positional embeddings."} + ) + checkpoint_dir: Optional[str] = field( + default=None, + metadata={"help": "Path to the directory(s) containing the delta model checkpoints as well as the configurations."} + ) + flash_attn: Optional[bool] = field( + default=False, + metadata={"help": "Enable FlashAttention-2 for faster training."} + ) + shift_attn: Optional[bool] = field( + default=False, + metadata={"help": "Enable shift short attention (S^2-Attn) proposed by LongLoRA."} + ) + reward_model: Optional[str] = field( + default=None, + metadata={"help": "Path to the directory containing the checkpoints of the reward model."} + ) + plot_loss: Optional[bool] = field( + default=False, + metadata={"help": "Whether to plot the training loss after fine-tuning or not."} + ) + hf_auth_token: Optional[str] = field( + default=None, + metadata={"help": "Auth token to log in with Hugging Face Hub."} + ) + export_dir: Optional[str] = field( + default=None, + metadata={"help": "Path to the directory to save the exported model."} + ) + + def __post_init__(self): + self.compute_dtype = None + self.model_max_length = None + + if self.split_special_tokens and self.use_fast_tokenizer: + raise ValueError("`split_special_tokens` is only supported for slow tokenizers.") + + if self.checkpoint_dir is not None: # support merging multiple lora weights + self.checkpoint_dir = [cd.strip() for cd in self.checkpoint_dir.split(",")] + + if self.quantization_bit is not None: + assert self.quantization_bit in [4, 8], "We only accept 4-bit or 8-bit quantization." + + if self.use_auth_token == True and self.hf_auth_token is not None: + from huggingface_hub.hf_api import HfFolder # lazy load + HfFolder.save_token(self.hf_auth_token) diff --git a/llm_rl/src/llmtuner/tuner/__init__.py b/llm_rl/src/llmtuner/tuner/__init__.py new file mode 100644 index 00000000..4d5a83e4 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/__init__.py @@ -0,0 +1 @@ +from llmtuner.tuner.tune import export_model, run_exp diff --git a/llm_rl/src/llmtuner/tuner/core/__init__.py b/llm_rl/src/llmtuner/tuner/core/__init__.py new file mode 100644 index 00000000..bd1c5cf0 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/core/__init__.py @@ -0,0 +1,2 @@ +from llmtuner.tuner.core.parser import get_train_args, get_infer_args +from llmtuner.tuner.core.loader import load_model_and_tokenizer diff --git a/llm_rl/src/llmtuner/tuner/core/adapter.py b/llm_rl/src/llmtuner/tuner/core/adapter.py new file mode 100644 index 00000000..4fcc6e62 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/core/adapter.py @@ -0,0 +1,101 @@ +import torch +from typing import TYPE_CHECKING + +from peft import ( + PeftModel, + TaskType, + LoraConfig, + get_peft_model +) + +from llmtuner.extras.logging import get_logger +from llmtuner.tuner.core.utils import find_all_linear_modules + +if TYPE_CHECKING: + from transformers.modeling_utils import PreTrainedModel + from llmtuner.hparams import ModelArguments, FinetuningArguments + + +logger = get_logger(__name__) + + +def init_adapter( + model: "PreTrainedModel", + model_args: "ModelArguments", + finetuning_args: "FinetuningArguments", + is_trainable: bool, + is_mergeable: bool +) -> "PreTrainedModel": + r""" + Initializes the adapters. + + Support full-parameter, freeze and LoRA training. + + Note that the trainable parameters must be cast to float32. + """ + + if finetuning_args.finetuning_type == "none" and is_trainable: + raise ValueError("You cannot use finetuning_type=none while training.") + + if finetuning_args.finetuning_type == "full" and is_trainable: + logger.info("Fine-tuning method: Full") + model = model.float() + + if finetuning_args.finetuning_type == "freeze": + logger.info("Fine-tuning method: Freeze") + num_layers = getattr(model.config, "num_layers") + if finetuning_args.num_layer_trainable > 0: # fine-tuning the last n layers if num_layer_trainable > 0 + trainable_layer_ids = [num_layers - k - 1 for k in range(finetuning_args.num_layer_trainable)] + else: # fine-tuning the first n layers if num_layer_trainable < 0 + trainable_layer_ids = [k for k in range(-finetuning_args.num_layer_trainable)] + + trainable_layers = ["{:d}.{}".format(idx, finetuning_args.name_module_trainable) for idx in trainable_layer_ids] + for name, param in model.named_parameters(): + if not any(trainable_layer in name for trainable_layer in trainable_layers): + param.requires_grad_(False) + else: + param.data = param.data.to(torch.float32) + + if finetuning_args.finetuning_type == "lora": + logger.info("Fine-tuning method: LoRA") + latest_checkpoint = None + + if model_args.checkpoint_dir is not None: + if (is_trainable and finetuning_args.resume_lora_training) or (not is_mergeable): # continually fine-tuning + checkpoints_to_merge, latest_checkpoint = model_args.checkpoint_dir[:-1], model_args.checkpoint_dir[-1] + else: + checkpoints_to_merge = model_args.checkpoint_dir + + for checkpoint in checkpoints_to_merge: + model = PeftModel.from_pretrained(model, checkpoint) + model = model.merge_and_unload() + + if len(checkpoints_to_merge) > 0: + logger.info("Merged {} model checkpoint(s).".format(len(checkpoints_to_merge))) + + if latest_checkpoint is not None: # resume lora training or quantized inference + model = PeftModel.from_pretrained(model, latest_checkpoint, is_trainable=is_trainable) + + if is_trainable and latest_checkpoint is None: # create new lora weights while training + if len(finetuning_args.lora_target) == 1 and finetuning_args.lora_target[0] == "all": + target_modules = find_all_linear_modules(model, model_args.quantization_bit) + else: + target_modules = finetuning_args.lora_target + + lora_config = LoraConfig( + task_type=TaskType.CAUSAL_LM, + inference_mode=False, + r=finetuning_args.lora_rank, + lora_alpha=finetuning_args.lora_alpha, + lora_dropout=finetuning_args.lora_dropout, + target_modules=target_modules, + modules_to_save=finetuning_args.additional_target + ) + model = get_peft_model(model, lora_config) + if id(model.peft_config) != id(model.base_model.peft_config): # https://github.com/huggingface/peft/issues/923 + model.base_model.peft_config = model.peft_config + + if model_args.checkpoint_dir is not None: + logger.info("Loaded fine-tuned model from checkpoint(s): {}".format(",".join(model_args.checkpoint_dir))) + + return model diff --git a/llm_rl/src/llmtuner/tuner/core/loader.py b/llm_rl/src/llmtuner/tuner/core/loader.py new file mode 100644 index 00000000..e77c4945 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/core/loader.py @@ -0,0 +1,244 @@ +import os +import math +import torch +from types import MethodType +from typing import TYPE_CHECKING, Literal, Optional, Tuple + +from transformers import ( + AutoConfig, + AutoModelForCausalLM, + AutoTokenizer, + BitsAndBytesConfig, + PretrainedConfig, + PreTrainedModel, + PreTrainedTokenizerBase +) +from transformers.models.llama import modeling_llama as LlamaModule +from transformers.utils.versions import require_version +from trl import AutoModelForCausalLMWithValueHead + +try: + from transformers.integrations import is_deepspeed_zero3_enabled +except ImportError: # https://github.com/huggingface/transformers/releases/tag/v4.33.1 + from transformers.deepspeed import is_deepspeed_zero3_enabled + +from llmtuner.extras.logging import reset_logging, get_logger +from llmtuner.extras.misc import count_parameters, infer_optim_dtype +from llmtuner.extras.patches import llama_patch as LlamaPatches +from llmtuner.extras.save_and_load import load_valuehead_params +from llmtuner.hparams import FinetuningArguments +from llmtuner.tuner.core.adapter import init_adapter +from llmtuner.tuner.core.utils import prepare_model_for_training + +if TYPE_CHECKING: + from transformers import PreTrainedTokenizer + from llmtuner.hparams import ModelArguments + + +logger = get_logger(__name__) + + +require_version("transformers>=4.31.0,<4.35.0", "To fix: pip install \"transformers>=4.31.0,<4.35.0\"") +require_version("datasets>=2.12.0", "To fix: pip install datasets>=2.12.0") +require_version("accelerate>=0.21.0", "To fix: pip install accelerate>=0.21.0") +require_version("peft>=0.4.0", "To fix: pip install peft>=0.4.0") +require_version("trl>=0.7.2", "To fix: pip install trl>=0.7.2") + + +def load_model_and_tokenizer( + model_args: "ModelArguments", + finetuning_args: "FinetuningArguments", + is_trainable: Optional[bool] = False, + stage: Optional[Literal["pt", "sft", "rm", "ppo"]] = "sft" +) -> Tuple[PreTrainedModel, "PreTrainedTokenizer"]: + r""" + Loads pretrained model and tokenizer. + + Support both training and inference. + """ + if (not is_trainable) and model_args.checkpoint_dir is None: + logger.warning("Checkpoint is not found at evaluation, load the original model.") + finetuning_args = FinetuningArguments(finetuning_type="none") + + config_kwargs = { + "trust_remote_code": True, + "cache_dir": model_args.cache_dir, + "revision": model_args.model_revision, + "use_auth_token": True if model_args.use_auth_token else None, + } + + tokenizer = AutoTokenizer.from_pretrained( + model_args.model_name_or_path, + use_fast=model_args.use_fast_tokenizer, + split_special_tokens=model_args.split_special_tokens, + padding_side="right", # training with left-padded tensors in fp16 precision may cause overflow + **config_kwargs + ) + + if finetuning_args.finetuning_type != "lora" and model_args.checkpoint_dir is not None: + model_to_load = model_args.checkpoint_dir[0] + else: + model_to_load = model_args.model_name_or_path + + config = AutoConfig.from_pretrained(model_to_load, **config_kwargs) + + # Fix tokenizer (for ChatGLM2 and ChatGLM3) + if getattr(config, "model_type", None) == "chatglm": + tokenizer._pad = MethodType(PreTrainedTokenizerBase._pad, tokenizer) + + # Set model dtype + if model_args.compute_dtype is not None: # for training + setattr(config, "torch_dtype", model_args.compute_dtype) + else: # for evaluation, priority: bf16 > fp16 > fp32 + model_args.compute_dtype = infer_optim_dtype(model_dtype=getattr(config, "torch_dtype", None)) + + # Fix config (for Qwen) + if getattr(config, "model_type", None) == "qwen": + for dtype_name, dtype in [("fp16", torch.float16), ("bf16", torch.bfloat16), ("fp32", torch.float32)]: + setattr(config, dtype_name, getattr(config, "torch_dtype", None) == dtype) + + # Set RoPE scaling + if model_args.rope_scaling is not None: + if hasattr(config, "use_dynamic_ntk"): # for Qwen models + if is_trainable: + logger.warning("Qwen model does not support RoPE scaling in training.") + else: + setattr(config, "use_dynamic_ntk", True) + setattr(config, "use_logn_attn", True) + logger.info("Using dynamic NTK scaling.") + + elif hasattr(config, "rope_scaling"): # for LLaMA and Falcon models + if is_trainable: + if model_args.rope_scaling == "dynamic": + logger.warning( + "Dynamic NTK may not work well with fine-tuning. " + "See: https://github.com/huggingface/transformers/pull/24653" + ) + + current_max_length = getattr(config, "max_position_embeddings", None) + if current_max_length and model_args.model_max_length > current_max_length: + scaling_factor = float(math.ceil(model_args.model_max_length / current_max_length)) + else: + logger.warning("Input length is smaller than max length. Consider increase input length.") + scaling_factor = 1.0 + else: + scaling_factor = 2.0 + + setattr(config, "rope_scaling", {"type": model_args.rope_scaling, "factor": scaling_factor}) + logger.info("Using {} scaling strategy and setting scaling factor to {}".format( + model_args.rope_scaling, scaling_factor + )) + + else: + logger.warning("Current model does not support RoPE scaling.") + + # Set FlashAttention-2 + if model_args.flash_attn: + if getattr(config, "model_type", None) == "llama": + LlamaModule.LlamaAttention = LlamaPatches.LlamaFlashAttention2 + LlamaModule.LlamaModel._prepare_decoder_attention_mask = LlamaPatches._prepare_decoder_attention_mask + logger.info("Using FlashAttention-2 for faster training and inference.") + elif getattr(config, "model_type", None) == "qwen": + logger.info("Qwen models automatically enable FlashAttention if installed.") + else: + logger.warning("Current model does not support FlashAttention-2.") + elif is_trainable and model_args.shift_attn and getattr(config, "model_type", None) == "llama": + LlamaModule.LlamaAttention = LlamaPatches.LlamaShiftShortAttention + logger.warning("Using `--flash_attn` for faster training in large context length.") + + # Set shift short attention (S^2-Attn) + if is_trainable and model_args.shift_attn: + if getattr(config, "model_type", None) == "llama": + setattr(config, "group_size_ratio", 0.25) + logger.info("Using shift short attention with group_size_ratio=1/4.") + else: + logger.warning("Current model does not support shift short attention.") + + # Quantization configurations (using bitsandbytes library). + is_mergeable = True + if model_args.quantization_bit is not None: + if is_deepspeed_zero3_enabled(): + raise ValueError("DeepSpeed ZeRO-3 is incompatible with quantization.") + + if model_args.quantization_bit == 8: + require_version("bitsandbytes>=0.37.0", "To fix: pip install bitsandbytes>=0.37.0") + config_kwargs["load_in_8bit"] = True + config_kwargs["quantization_config"] = BitsAndBytesConfig(load_in_8bit=True) + + elif model_args.quantization_bit == 4: + require_version("bitsandbytes>=0.39.0", "To fix: pip install bitsandbytes>=0.39.0") + config_kwargs["load_in_4bit"] = True + config_kwargs["quantization_config"] = BitsAndBytesConfig( + load_in_4bit=True, + bnb_4bit_compute_dtype=model_args.compute_dtype, + bnb_4bit_use_double_quant=model_args.double_quantization, + bnb_4bit_quant_type=model_args.quantization_type + ) + + is_mergeable = False + config_kwargs["device_map"] = {"": int(os.environ.get("LOCAL_RANK", "0"))} if is_trainable else "auto" + logger.info("Quantizing model to {} bit.".format(model_args.quantization_bit)) + + # Load and prepare pre-trained models (without valuehead). + model = AutoModelForCausalLM.from_pretrained( + model_to_load, + config=config, + torch_dtype=model_args.compute_dtype, + low_cpu_mem_usage=(not is_deepspeed_zero3_enabled()), + **config_kwargs + ) + + # Disable custom generate method (for Qwen and Baichuan2) + if isinstance(model, PreTrainedModel) and "GenerationMixin" not in str(model.generate.__func__): + model.generate = MethodType(PreTrainedModel.generate, model) + + # Fix LM head (for ChatGLM2 and ChatGLM3) + if getattr(config, "model_type", None) == "chatglm": + setattr(model, "lm_head", model.transformer.output_layer) + setattr(model, "_keys_to_ignore_on_save", ["lm_head.weight"]) + + # Register auto class to save the custom code files. + if isinstance(config, PretrainedConfig) and "AutoConfig" in getattr(config, "auto_map", {}): + config.__class__.register_for_auto_class() + if isinstance(model, PreTrainedModel) and "AutoModelForCausalLM" in getattr(config, "auto_map", {}): + model.__class__.register_for_auto_class() + if isinstance(tokenizer, PreTrainedTokenizerBase) and "AutoTokenizer" in tokenizer.init_kwargs.get("auto_map", {}): + tokenizer.__class__.register_for_auto_class() + + # Initialize adapters + model = prepare_model_for_training(model=model, finetuning_args=finetuning_args) if is_trainable else model + model = init_adapter(model, model_args, finetuning_args, is_trainable, is_mergeable) + model = model.train() if is_trainable else model.eval() + + # Prepare model with valuehead for RLHF + if stage == "rm" or stage == "ppo": + model: "AutoModelForCausalLMWithValueHead" = AutoModelForCausalLMWithValueHead.from_pretrained(model) + reset_logging() + if stage == "rm" and model_args.checkpoint_dir is not None: # load valuehead weights to evaluate reward model + logger.warning("Only the last checkpoint containing valuehead will be loaded.") + if load_valuehead_params(model, model_args.checkpoint_dir[-1]): + model.v_head.load_state_dict({ + "summary.weight": getattr(model, "reward_head_weight"), + "summary.bias": getattr(model, "reward_head_bias") + }) + + if stage == "ppo": # load reward model + logger.info("Load reward model from {}".format(model_args.reward_model)) + if getattr(model, "is_peft_model", False): + model.pretrained_model.load_adapter(model_args.reward_model, "reward") + assert load_valuehead_params(model, model_args.reward_model), "Reward model is not correctly loaded." + + # Prepare model for inference + if not is_trainable: + model.requires_grad_(False) # fix all model params + model = model.to(model_args.compute_dtype) if model_args.quantization_bit is None else model + + trainable_params, all_param = count_parameters(model) + logger.info("trainable params: {:d} || all params: {:d} || trainable%: {:.4f}".format( + trainable_params, all_param, 100 * trainable_params / all_param + )) + + if not is_trainable: + logger.info("This IS expected that the trainable params is 0 if you are using model for inference only.") + + return model, tokenizer diff --git a/llm_rl/src/llmtuner/tuner/core/parser.py b/llm_rl/src/llmtuner/tuner/core/parser.py new file mode 100644 index 00000000..603fc1bc --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/core/parser.py @@ -0,0 +1,226 @@ +import os +import sys +import torch +import datasets +import transformers +from typing import Any, Dict, Optional, Tuple +from transformers import HfArgumentParser, Seq2SeqTrainingArguments +from transformers.trainer_utils import get_last_checkpoint + +from llmtuner.extras.logging import get_logger +from llmtuner.hparams import ( + ModelArguments, + DataArguments, + FinetuningArguments, + GeneratingArguments +) + + +logger = get_logger(__name__) + + +def _parse_args(parser: HfArgumentParser, args: Optional[Dict[str, Any]] = None) -> Tuple[Any]: + if args is not None: + return parser.parse_dict(args) + elif len(sys.argv) == 2 and sys.argv[1].endswith(".yaml"): + return parser.parse_yaml_file(os.path.abspath(sys.argv[1])) + elif len(sys.argv) == 2 and sys.argv[1].endswith(".json"): + return parser.parse_json_file(os.path.abspath(sys.argv[1])) + else: + return parser.parse_args_into_dataclasses() + + +def parse_train_args( + args: Optional[Dict[str, Any]] = None +) -> Tuple[ + ModelArguments, + DataArguments, + Seq2SeqTrainingArguments, + FinetuningArguments, + GeneratingArguments +]: + parser = HfArgumentParser(( + ModelArguments, + DataArguments, + Seq2SeqTrainingArguments, + FinetuningArguments, + GeneratingArguments + )) + return _parse_args(parser, args) + + +def parse_infer_args( + args: Optional[Dict[str, Any]] = None +) -> Tuple[ + ModelArguments, + DataArguments, + FinetuningArguments, + GeneratingArguments +]: + parser = HfArgumentParser(( + ModelArguments, + DataArguments, + FinetuningArguments, + GeneratingArguments + )) + return _parse_args(parser, args) + + +def get_train_args( + args: Optional[Dict[str, Any]] = None +) -> Tuple[ + ModelArguments, + DataArguments, + Seq2SeqTrainingArguments, + FinetuningArguments, + GeneratingArguments +]: + model_args, data_args, training_args, finetuning_args, generating_args = parse_train_args(args) + + # Setup logging + if training_args.should_log: + # The default of training_args.log_level is passive, so we set log level at info here to have that default. + transformers.utils.logging.set_verbosity_info() + + log_level = training_args.get_process_log_level() + datasets.utils.logging.set_verbosity(log_level) + transformers.utils.logging.set_verbosity(log_level) + transformers.utils.logging.enable_default_handler() + transformers.utils.logging.enable_explicit_format() + + # Check arguments + data_args.init_for_training(training_args.seed) + + if finetuning_args.stage != "pt" and data_args.template is None: + raise ValueError("Please specify which `template` to use.") + + if finetuning_args.stage != "sft" and training_args.predict_with_generate: + raise ValueError("`predict_with_generate` cannot be set as True except SFT.") + + if finetuning_args.stage == "sft" and training_args.do_predict and not training_args.predict_with_generate: + raise ValueError("Please enable `predict_with_generate` to save model predictions.") + + if finetuning_args.stage in ["rm", "ppo"] and finetuning_args.finetuning_type != "lora": + raise ValueError("RM and PPO stages can only be performed with the LoRA method.") + + if finetuning_args.stage in ["rm", "ppo"] and training_args.resume_from_checkpoint is not None: + raise ValueError("RM and PPO stages do not support `resume_from_checkpoint`.") + + if finetuning_args.stage == "ppo" and not training_args.do_train: + raise ValueError("PPO training does not support evaluation.") + + if finetuning_args.stage in ["rm", "dpo"]: + for dataset_attr in data_args.dataset_list: + if not dataset_attr.ranking: + raise ValueError("Please use ranked datasets for reward modeling or DPO training.") + + if finetuning_args.stage == "ppo" and model_args.reward_model is None: + raise ValueError("Reward model is necessary for PPO training.") + + if finetuning_args.stage == "ppo" and model_args.shift_attn: + raise ValueError("PPO training is incompatible with S^2-Attn.") + + if training_args.max_steps == -1 and data_args.streaming: + raise ValueError("Please specify `max_steps` in streaming mode.") + + if training_args.do_train and training_args.predict_with_generate: + raise ValueError("`predict_with_generate` cannot be set as True while training.") + + if training_args.do_train and finetuning_args.finetuning_type == "lora" and finetuning_args.lora_target is None: + raise ValueError("Please specify `lora_target` in LoRA training.") + + if model_args.quantization_bit is not None and finetuning_args.finetuning_type != "lora": + raise ValueError("Quantization is only compatible with the LoRA method.") + + if model_args.checkpoint_dir is not None: + if finetuning_args.finetuning_type != "lora" and len(model_args.checkpoint_dir) != 1: + raise ValueError("Only LoRA tuning accepts multiple checkpoints.") + + if model_args.quantization_bit is not None: + if len(model_args.checkpoint_dir) != 1: + raise ValueError("Quantized model only accepts a single checkpoint. Merge them first.") + + if not finetuning_args.resume_lora_training: + raise ValueError("Quantized model cannot create new LoRA weight. Merge them first.") + + if training_args.do_train and model_args.quantization_bit is not None and (not finetuning_args.upcast_layernorm): + logger.warning("We recommend enable `upcast_layernorm` in quantized training.") + + if training_args.do_train and (not training_args.fp16) and (not training_args.bf16): + logger.warning("We recommend enable mixed precision training.") + + if (not training_args.do_train) and model_args.quantization_bit is not None: + logger.warning("Evaluating model in 4/8-bit mode may cause lower scores.") + + # postprocess training_args + if ( + training_args.local_rank != -1 + and training_args.ddp_find_unused_parameters is None + and finetuning_args.finetuning_type == "lora" + ): + logger.warning("`ddp_find_unused_parameters` needs to be set as False for LoRA in DDP training.") + training_args_dict = training_args.to_dict() + training_args_dict.update(dict(ddp_find_unused_parameters=False)) + training_args = Seq2SeqTrainingArguments(**training_args_dict) + + if ( + training_args.resume_from_checkpoint is None + and training_args.do_train + and os.path.isdir(training_args.output_dir) + and not training_args.overwrite_output_dir + ): + last_checkpoint = get_last_checkpoint(training_args.output_dir) + if last_checkpoint is None and len(os.listdir(training_args.output_dir)) > 0: + raise ValueError("Output directory already exists and is not empty. Please set `overwrite_output_dir`.") + + if last_checkpoint is not None: + training_args_dict = training_args.to_dict() + training_args_dict.update(dict(resume_from_checkpoint=last_checkpoint)) + training_args = Seq2SeqTrainingArguments(**training_args_dict) + logger.info( + "Resuming from checkpoint. Change `output_dir` or use `overwrite_output_dir` to avoid." + ) + + # postprocess model_args + model_args.compute_dtype = ( + torch.bfloat16 if training_args.bf16 else (torch.float16 if training_args.fp16 else None) + ) + model_args.model_max_length = data_args.cutoff_len + + # Log on each process the small summary: + logger.info("Process rank: {}, device: {}, n_gpu: {}\n distributed training: {}, compute dtype: {}".format( + training_args.local_rank, training_args.device, training_args.n_gpu, + bool(training_args.local_rank != -1), str(model_args.compute_dtype) + )) + logger.info(f"Training/evaluation parameters {training_args}") + + # Set seed before initializing model. + transformers.set_seed(training_args.seed) + + return model_args, data_args, training_args, finetuning_args, generating_args + + +def get_infer_args( + args: Optional[Dict[str, Any]] = None +) -> Tuple[ + ModelArguments, + DataArguments, + FinetuningArguments, + GeneratingArguments +]: + model_args, data_args, finetuning_args, generating_args = parse_infer_args(args) + + if data_args.template is None: + raise ValueError("Please specify which `template` to use.") + + if model_args.quantization_bit is not None and finetuning_args.finetuning_type != "lora": + raise ValueError("Quantization is only compatible with the LoRA method.") + + if model_args.checkpoint_dir is not None: + if finetuning_args.finetuning_type != "lora" and len(model_args.checkpoint_dir) != 1: + raise ValueError("Only LoRA tuning accepts multiple checkpoints.") + + if model_args.quantization_bit is not None and len(model_args.checkpoint_dir) != 1: + raise ValueError("Quantized model only accepts a single checkpoint. Merge them first.") + + return model_args, data_args, finetuning_args, generating_args diff --git a/llm_rl/src/llmtuner/tuner/core/utils.py b/llm_rl/src/llmtuner/tuner/core/utils.py new file mode 100644 index 00000000..d9a1aac9 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/core/utils.py @@ -0,0 +1,94 @@ +import torch +from types import MethodType +from typing import TYPE_CHECKING, List, Optional + +from llmtuner.extras.constants import LAYERNORM_NAMES +from llmtuner.extras.logging import get_logger + +if TYPE_CHECKING: + from transformers.modeling_utils import PreTrainedModel + from llmtuner.hparams import FinetuningArguments + + +logger = get_logger(__name__) + + +def find_all_linear_modules( + model: "PreTrainedModel", + quantization_bit: Optional[int] = None, + output_layer_name: Optional[str] = "lm_head" +) -> List[str]: + if quantization_bit is not None: + import bitsandbytes as bnb + linear_cls = bnb.nn.Linear4bit if quantization_bit == 4 else bnb.nn.Linear8bitLt + else: + linear_cls = torch.nn.Linear + + module_names = set() + for name, module in model.named_modules(): + if output_layer_name not in name and isinstance(module, linear_cls): + module_names.add(name.split(".")[-1]) + + if output_layer_name in module_names: + module_names.pop(output_layer_name) + + return list(module_names) + + +def prepare_model_for_training( + model: "PreTrainedModel", + finetuning_args: "FinetuningArguments", + output_layer_name: Optional[str] = "lm_head", + use_gradient_checkpointing: Optional[bool] = True, + layernorm_names: Optional[List[str]] = LAYERNORM_NAMES +) -> "PreTrainedModel": + r""" + Includes: + (1) cast the layernorm in fp32 + (2) make output embedding layer require grads + (3) upcast the lm_head to fp32 + Inspired by: https://github.com/huggingface/peft/blob/v0.2.0/src/peft/utils/other.py#L33 + """ + if finetuning_args.upcast_layernorm: + for name, param in model.named_parameters(): + if param.ndim == 1 and any(ln_name in name for ln_name in layernorm_names): + param.data = param.data.to(torch.float32) + logger.info("Upcasting weights in layernorm in float32.") + + if finetuning_args.neft_alpha > 1e-6: + input_embed = model.get_input_embeddings() + if isinstance(input_embed, torch.nn.Embedding): + def noisy_forward(self: torch.nn.Embedding, x: torch.Tensor) -> torch.Tensor: + embeddings = input_embed.__class__.forward(self, x) + if self.training: + dims = self.num_embeddings * self.embedding_dim + mag_norm = finetuning_args.neft_alpha / (dims ** 0.5) + embeddings += torch.zeros_like(embeddings).uniform_(-mag_norm, mag_norm) + return embeddings + + input_embed.forward = MethodType(noisy_forward, input_embed) + logger.info("Using noisy embedding with alpha={:.2f}".format(finetuning_args.neft_alpha)) + else: + logger.warning("Input embeddings are not normal nn.Embedding, cannot transform into noisy embedding.") + + if use_gradient_checkpointing: + if hasattr(model, "enable_input_require_grads"): + model.enable_input_require_grads() + else: + def make_inputs_require_grad(module: torch.nn.Module, input: torch.Tensor, output: torch.Tensor): + output.requires_grad_(True) + model.get_input_embeddings().register_forward_hook(make_inputs_require_grad) + + model.gradient_checkpointing_enable() + model.config.use_cache = False # turn off when gradient checkpointing is enabled + logger.info("Gradient checkpointing enabled.") + + if finetuning_args.finetuning_type != "full" and hasattr(model, output_layer_name): + output_layer = getattr(model, output_layer_name) + if isinstance(output_layer, torch.nn.Linear): + def forward_in_fp32(self, x: torch.Tensor) -> torch.Tensor: + return output_layer.__class__.forward(self, x.to(output_layer.weight.dtype)).to(torch.float32) + + output_layer.forward = MethodType(forward_in_fp32, output_layer) + + return model diff --git a/llm_rl/src/llmtuner/tuner/dpo/__init__.py b/llm_rl/src/llmtuner/tuner/dpo/__init__.py new file mode 100644 index 00000000..f2b5cfb5 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/dpo/__init__.py @@ -0,0 +1 @@ +from llmtuner.tuner.dpo.workflow import run_dpo diff --git a/llm_rl/src/llmtuner/tuner/dpo/collator.py b/llm_rl/src/llmtuner/tuner/dpo/collator.py new file mode 100644 index 00000000..5c862b4f --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/dpo/collator.py @@ -0,0 +1,51 @@ +import torch +from dataclasses import dataclass +from typing import Any, Dict, List, Sequence, Tuple +from transformers import DataCollatorForSeq2Seq + + +@dataclass +class DPODataCollatorWithPadding(DataCollatorForSeq2Seq): + r""" + Data collator for pairwise data. + """ + + def _pad_labels(self, batch: torch.Tensor, positions: List[Tuple[int, int]]) -> torch.Tensor: + padded_labels = [] + for feature, (prompt_len, answer_len) in zip(batch, positions): + if self.tokenizer.padding_side == "left": + start, end = feature.size(0) - answer_len, feature.size(0) + else: + start, end = prompt_len, prompt_len + answer_len + padded_tensor = self.label_pad_token_id * torch.ones_like(feature) + padded_tensor[start:end] = feature[start:end] + padded_labels.append(padded_tensor) + return torch.stack(padded_labels, dim=0).contiguous() # in contiguous memory + + def __call__(self, features: Sequence[Dict[str, Any]]) -> Dict[str, torch.Tensor]: + r""" + Pads batched data to the longest sequence in the batch. + + We generate 2 * n examples where the first n examples represent chosen examples and + the last n examples represent rejected examples. + """ + concatenated_features = [] + label_positions = [] + for key in ("chosen_ids", "rejected_ids"): + for feature in features: + prompt_len, answer_len = len(feature["prompt_ids"]), len(feature[key]) + concatenated_features.append({ + "input_ids": feature["prompt_ids"] + feature[key], + "attention_mask": [1] * (prompt_len + answer_len) + }) + label_positions.append((prompt_len, answer_len)) + + batch = self.tokenizer.pad( + concatenated_features, + padding=self.padding, + max_length=self.max_length, + pad_to_multiple_of=self.pad_to_multiple_of, + return_tensors=self.return_tensors, + ) + batch["labels"] = self._pad_labels(batch["input_ids"], label_positions) + return batch diff --git a/llm_rl/src/llmtuner/tuner/dpo/trainer.py b/llm_rl/src/llmtuner/tuner/dpo/trainer.py new file mode 100644 index 00000000..8a9f8dd6 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/dpo/trainer.py @@ -0,0 +1,104 @@ +import torch +import deepspeed # type: ignore +from copy import deepcopy +from collections import defaultdict +from typing import TYPE_CHECKING, Dict, Literal, Optional, Tuple, Union +from transformers import BatchEncoding, Trainer +from trl import DPOTrainer +from trl.trainer.utils import disable_dropout_in_model + +from llmtuner.extras.constants import IGNORE_INDEX + +if TYPE_CHECKING: + from transformers import PreTrainedModel + from trl import PreTrainedModelWrapper + + +class CustomDPOTrainer(DPOTrainer): + + def __init__( + self, + beta: float, + model: Union["PreTrainedModel", torch.nn.Module], + ref_model: Optional[Union["PreTrainedModel", torch.nn.Module]] = None, + disable_dropout: Optional[bool] = True, + loss_type: Optional[Literal["sigmoid", "hinge"]] = "sigmoid", + **kwargs + ): + if disable_dropout: + disable_dropout_in_model(model) + if ref_model is not None: + disable_dropout_in_model(ref_model) + + self.is_encoder_decoder = model.config.is_encoder_decoder + self.ref_model = ref_model + self.use_dpo_data_collator = True # hack to avoid warning + self.generate_during_eval = False # disable at evaluation + self.label_pad_token_id = IGNORE_INDEX + self.padding_value = 0 + self.beta = beta + self.loss_type = loss_type + self._stored_metrics = defaultdict(lambda: defaultdict(list)) + + Trainer.__init__(self, model=model, **kwargs) + if not hasattr(self, "accelerator"): + raise AttributeError("Please update `transformers`.") + + if ref_model is not None: + if self.is_deepspeed_enabled: + self.ref_model = self._prepare_deepspeed(self.ref_model) + else: + self.ref_model = self.accelerator.prepare_model(self.ref_model, evaluation_mode=True) + + def _prepare_deepspeed(self, model: "PreTrainedModelWrapper"): + # Adapted from accelerate: https://github.com/huggingface/accelerate/blob/739b135f8367becb67ffaada12fe76e3aa60fefd/src/accelerate/accelerator.py#L1473 + deepspeed_plugin = self.accelerator.state.deepspeed_plugin + config_kwargs = deepcopy(deepspeed_plugin.deepspeed_config) + if model is not None: + if hasattr(model, "config"): + hidden_size = ( + max(model.config.hidden_sizes) + if getattr(model.config, "hidden_sizes", None) + else getattr(model.config, "hidden_size", None) + ) + if hidden_size is not None and config_kwargs["zero_optimization"]["stage"] == 3: + # Note that `stage3_prefetch_bucket_size` can produce DeepSpeed messages like: `Invalidate trace cache @ step 0: expected module 1, but got module 0` + # This is expected and is not an error, see: https://github.com/microsoft/DeepSpeed/discussions/4081 + config_kwargs.update( + { + "zero_optimization.reduce_bucket_size": hidden_size * hidden_size, + "zero_optimization.stage3_param_persistence_threshold": 10 * hidden_size, + "zero_optimization.stage3_prefetch_bucket_size": 0.9 * hidden_size * hidden_size, + } + ) + + # If ZeRO-3 is used, we shard both the active and reference model. + # Otherwise, we assume the reference model fits in memory and is initialized on each device with ZeRO disabled (stage 0) + if config_kwargs["zero_optimization"]["stage"] != 3: + config_kwargs["zero_optimization"]["stage"] = 0 + model, *_ = deepspeed.initialize(model=model, config=config_kwargs) + model.eval() + return model + + def concatenated_forward( + self, + model: Optional[torch.nn.Module] = None, + batch: Optional[Dict[str, torch.Tensor]] = None + ) -> Tuple[torch.FloatTensor, torch.FloatTensor, torch.FloatTensor, torch.FloatTensor]: + batch_copied = BatchEncoding({k: v.detach().clone() for k, v in batch.items()}) # avoid error + + all_logits = model( + input_ids=batch_copied["input_ids"], + attention_mask=batch_copied["attention_mask"], + return_dict=True + ).logits.to(torch.float32) + + all_logps = self._get_batch_logps( + all_logits, + batch["labels"], + average_log_prob=False + ) + batch_size = batch["input_ids"].size(0) // 2 + chosen_logps, rejected_logps = all_logps.split(batch_size, dim=0) + chosen_logits, rejected_logits = all_logits.split(batch_size, dim=0) + return chosen_logps, rejected_logps, chosen_logits, rejected_logits diff --git a/llm_rl/src/llmtuner/tuner/dpo/workflow.py b/llm_rl/src/llmtuner/tuner/dpo/workflow.py new file mode 100644 index 00000000..6e16dd18 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/dpo/workflow.py @@ -0,0 +1,66 @@ +# Inspired by: https://github.com/huggingface/trl/blob/main/examples/research_projects/stack_llama_2/scripts/dpo_llama2.py + +from copy import deepcopy +from peft import PeftModel +from typing import TYPE_CHECKING, Optional, List +from transformers import Seq2SeqTrainingArguments + +from llmtuner.dsets import get_dataset, preprocess_dataset, split_dataset +from llmtuner.extras.constants import IGNORE_INDEX +from llmtuner.extras.ploting import plot_loss +from llmtuner.tuner.core import load_model_and_tokenizer +from llmtuner.tuner.dpo.collator import DPODataCollatorWithPadding +from llmtuner.tuner.dpo.trainer import CustomDPOTrainer + +if TYPE_CHECKING: + from transformers import TrainerCallback + from llmtuner.hparams import ModelArguments, DataArguments, FinetuningArguments + + +def run_dpo( + model_args: "ModelArguments", + data_args: "DataArguments", + training_args: "Seq2SeqTrainingArguments", + finetuning_args: "FinetuningArguments", + callbacks: Optional[List["TrainerCallback"]] = None +): + dataset = get_dataset(model_args, data_args) + model, tokenizer = load_model_and_tokenizer(model_args, finetuning_args, training_args.do_train, stage="sft") + dataset = preprocess_dataset(dataset, tokenizer, data_args, training_args, stage="rm") + data_collator = DPODataCollatorWithPadding( + tokenizer=tokenizer, + pad_to_multiple_of=4, + label_pad_token_id=IGNORE_INDEX if data_args.ignore_pad_token_for_loss else tokenizer.pad_token_id + ) + + training_args_dict = training_args.to_dict() + training_args_dict.update(dict(remove_unused_columns=False)) # important for pairwise dataset + training_args = Seq2SeqTrainingArguments(**training_args_dict) + + # Initialize our Trainer + trainer = CustomDPOTrainer( + beta=finetuning_args.dpo_beta, + model=model, + ref_model=deepcopy(model) if not isinstance(model, PeftModel) else None, + args=training_args, + tokenizer=tokenizer, + data_collator=data_collator, + callbacks=callbacks, + **split_dataset(dataset, data_args, training_args) + ) + + # Training + if training_args.do_train: + train_result = trainer.train(resume_from_checkpoint=training_args.resume_from_checkpoint) + trainer.log_metrics("train", train_result.metrics) + trainer.save_metrics("train", train_result.metrics) + trainer.save_state() + trainer.save_model() + if trainer.is_world_process_zero() and model_args.plot_loss: + plot_loss(training_args.output_dir, keys=["loss", "eval_loss"]) + + # Evaluation + if training_args.do_eval: + metrics = trainer.evaluate(metric_key_prefix="eval") + trainer.log_metrics("eval", metrics) + trainer.save_metrics("eval", metrics) diff --git a/llm_rl/src/llmtuner/tuner/ppo/__init__.py b/llm_rl/src/llmtuner/tuner/ppo/__init__.py new file mode 100644 index 00000000..11519bab --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/ppo/__init__.py @@ -0,0 +1 @@ +from llmtuner.tuner.ppo.workflow import run_ppo diff --git a/llm_rl/src/llmtuner/tuner/ppo/trainer.py b/llm_rl/src/llmtuner/tuner/ppo/trainer.py new file mode 100644 index 00000000..372c4891 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/ppo/trainer.py @@ -0,0 +1,310 @@ +import os +import sys +import math +import torch +from tqdm import tqdm +from typing import TYPE_CHECKING, Any, Dict, List, Optional, Tuple + +from transformers import GenerationConfig, Trainer, TrainerState, TrainerControl +from transformers.trainer_utils import PREFIX_CHECKPOINT_DIR + +from trl import PPOTrainer +from trl.core import PPODecorators, logprobs_from_logits + +from llmtuner.extras.callbacks import LogCallback, SavePeftModelCallback +from llmtuner.extras.logging import get_logger +from llmtuner.extras.misc import AverageMeter, count_parameters, get_logits_processor +from llmtuner.tuner.ppo.utils import dump_layernorm, restore_layernorm, replace_model + +if TYPE_CHECKING: + from transformers import Seq2SeqTrainingArguments, TrainerCallback + from trl import AutoModelForCausalLMWithValueHead + from llmtuner.hparams import ModelArguments, FinetuningArguments, GeneratingArguments + + +logger = get_logger(__name__) + + +class CustomPPOTrainer(PPOTrainer, Trainer): + r""" + Inherits PPOTrainer. + """ + + def __init__( + self, + model_args: "ModelArguments", + training_args: "Seq2SeqTrainingArguments", + finetuning_args: "FinetuningArguments", + generating_args: "GeneratingArguments", + callbacks: List["TrainerCallback"], + **kwargs + ): + PPOTrainer.__init__(self, **kwargs) + self.args = training_args + self.model_args = model_args + self.finetuning_args = finetuning_args + self.generation_config = GenerationConfig( + pad_token_id=self.tokenizer.pad_token_id, + eos_token_id=[self.tokenizer.eos_token_id] + self.tokenizer.additional_special_tokens_ids, + **generating_args.to_dict() + ) + self.state = TrainerState() + self.control = TrainerControl() + self.log_callback, self.save_callback = callbacks[0], callbacks[1] + assert isinstance(self.log_callback, LogCallback) and isinstance(self.save_callback, SavePeftModelCallback) + if self.args.max_steps > 0: + logger.info("max_steps is given, it will override any value given in num_train_epochs") + + def ppo_train(self) -> None: + r""" + Implements training loop for the PPO stage, like _inner_training_loop() in Huggingface's Trainer. + """ + total_train_batch_size = ( + self.args.per_device_train_batch_size * self.args.gradient_accumulation_steps * self.args.world_size + ) + if self.args.max_steps > 0: + num_examples = total_train_batch_size * self.args.max_steps + num_train_epochs = sys.maxsize + max_steps = self.args.max_steps + steps_in_epoch = self.args.max_steps * self.args.gradient_accumulation_steps + else: + len_dataloader = len(self.dataloader) + num_examples = len(self.dataset) + num_train_epochs = self.args.num_train_epochs + max_steps = math.ceil(num_train_epochs * len_dataloader) + steps_in_epoch = len_dataloader + + self.state.max_steps = max_steps + self.state.num_train_epochs = num_train_epochs + self.state.is_local_process_zero = self.is_local_process_zero() + self.state.is_world_process_zero = self.is_world_process_zero() + + if self.is_world_process_zero(): + logger.info("***** Running training *****") + logger.info(f" Num examples = {num_examples}") + logger.info(f" Num Epochs = {num_train_epochs}") + logger.info(f" Instantaneous batch size per device = {self.args.per_device_train_batch_size}") + logger.info(f" Total train batch size (w. parallel, distributed & accumulation) = {total_train_batch_size}") + logger.info(f" Gradient Accumulation steps = {self.args.gradient_accumulation_steps}") + logger.info(f" Total optimization steps = {max_steps}") + logger.info(f" Number of trainable parameters = {count_parameters(self.model)[0]}") + + unwrapped_model: "AutoModelForCausalLMWithValueHead" = self.accelerator.unwrap_model(self.model) + dataiter = iter(self.dataloader) + loss_meter = AverageMeter() + reward_meter = AverageMeter() + self.log_callback.on_train_begin(self.args, self.state, self.control) + + for step in tqdm(range(max_steps), disable=not self.is_local_process_zero()): + try: + batch = next(dataiter) + except StopIteration: + dataiter = iter(self.dataloader) + batch = next(dataiter) + + # Cast to inference mode + unwrapped_model.gradient_checkpointing_disable() + unwrapped_model.config.use_cache = True + self.model.eval() + + # Get inputs + queries, responses = self.get_inputs(batch) + self.tokenizer.padding_side = "right" # change padding side + rewards = self.get_rewards(queries, responses, unwrapped_model) + + # Cast to training mode + unwrapped_model.gradient_checkpointing_enable() + unwrapped_model.config.use_cache = False + self.model.train() + + # Run PPO step + stats = self.step(queries, responses, rewards) + self.tokenizer.padding_side = "left" # restore padding side + loss_meter.update(float(stats["ppo/loss/total"]), n=len(rewards)) + reward_meter.update(torch.stack(rewards).mean().item(), n=len(rewards)) + + if self.config.log_with is not None: + try: + batch["query"] = self.tokenizer.batch_decode(queries, skip_special_tokens=True) + batch["response"] = self.tokenizer.batch_decode(responses, skip_special_tokens=True) + self.log_stats(stats, batch, rewards) + except: + logger.warning("Failed to save stats due to unknown errors.") + + self.state.global_step += 1 + self.log_callback.on_step_end(self.args, self.state, self.control) + + if self.is_local_process_zero() and (step+1) % self.args.logging_steps == 0: + logs = dict( + loss=round(loss_meter.avg, 4), + reward=round(reward_meter.avg, 4), + learning_rate=stats["ppo/learning_rate"], + epoch=round(step / steps_in_epoch, 2) + ) + tqdm.write(str(logs)) + logs["step"] = step + self.state.log_history.append(logs) + self.log_callback.on_log(self.args, self.state, self.control) + loss_meter.reset() + reward_meter.reset() + + if (step+1) % self.args.save_steps == 0: # save checkpoint + self.save_model(os.path.join( + self.args.output_dir, "{}-{}".format(PREFIX_CHECKPOINT_DIR, self.state.global_step) + )) + self.save_callback.on_save( + self.args, self.state, self.control, model=self.accelerator.unwrap_model(self.model) + ) + + if self.control.should_epoch_stop or self.control.should_training_stop: + break + + self.log_callback.on_train_end(self.args, self.state, self.control) + self.save_callback.on_train_end( + self.args, self.state, self.control, model=self.accelerator.unwrap_model(self.model) + ) + + @torch.no_grad() + def get_inputs(self, batch: Dict[str, torch.Tensor]) -> Tuple[List[torch.Tensor], List[torch.Tensor]]: + r""" + Generates model's responses given queries. + """ + if self.finetuning_args.upcast_layernorm: + layernorm_params = dump_layernorm(self.model) + + unwrapped_model: "AutoModelForCausalLMWithValueHead" = self.accelerator.unwrap_model(self.model) + response: torch.Tensor = unwrapped_model.generate( + generation_config=self.generation_config, + logits_processor=get_logits_processor(), + **batch + ) + + if self.finetuning_args.upcast_layernorm: + restore_layernorm(self.model, layernorm_params) + + query, response = batch["input_ids"].detach().cpu(), response[:, batch["input_ids"].size(-1):].detach().cpu() + queries, responses = [], [] + for i in range(len(query)): + query_length = (query[i] != self.tokenizer.pad_token_id).nonzero()[0].item() + response_index = (response[i] != self.tokenizer.pad_token_id).nonzero() + + if len(response_index) == 0: + response_length = 1 # allow empty response + elif self.tokenizer.pad_token_id == self.tokenizer.eos_token_id: + response_length = response_index[-1].item() + 2 # save the EOS token + else: + response_length = response_index[-1].item() + 1 + + queries.append(query[i, query_length:]) # remove padding from left + responses.append(response[i, :response_length]) # remove padding from right + + return queries, responses + + @torch.no_grad() + def get_rewards( + self, + queries: List[torch.Tensor], + responses: List[torch.Tensor], + unwrapped_model: "AutoModelForCausalLMWithValueHead" + ) -> List[torch.Tensor]: + r""" + Computes scores using given reward model. + """ + replace_model(unwrapped_model, target="reward") + batch = self.prepare_model_inputs(queries, responses) + + with torch.cuda.amp.autocast(dtype=self.model_args.compute_dtype): # support bf16 + _, _, values = self.model(**batch, output_hidden_states=True, return_dict=True) + + if values.size(0) != batch["input_ids"].size(0): # adapt to chatglm2 + values = torch.transpose(values, 0, 1) + + rewards = [] + for i in range(values.size(0)): + end_index = batch["attention_mask"][i].nonzero()[-1].item() # use the score on the EOS token + rewards.append(values[i, end_index].float().detach().cpu()) # use fp32 type + + replace_model(unwrapped_model, target="default") + return rewards + + @PPODecorators.empty_cuda_cache() + def batched_forward_pass( + self, + model: "AutoModelForCausalLMWithValueHead", + queries: torch.Tensor, + responses: torch.Tensor, + model_inputs: dict, + return_logits: Optional[bool] = False, + response_masks: Optional[torch.Tensor] = None + ): + r""" + Calculates model outputs in multiple batches. + + Subclass and override to inject custom behavior. + """ + bs = len(queries) + fbs = self.config.mini_batch_size + all_logprobs = [] + all_logits = [] + all_masks = [] + all_values = [] + + for i in range(math.ceil(bs / fbs)): + input_kwargs = {key: value[i * fbs : (i + 1) * fbs] for key, value in model_inputs.items()} + query_batch = queries[i * fbs : (i + 1) * fbs] + response_batch = responses[i * fbs : (i + 1) * fbs] + if response_masks is not None: + response_masks_batch = response_masks[i * fbs : (i + 1) * fbs] + input_ids = input_kwargs["input_ids"] + attention_mask = input_kwargs["attention_mask"] + + with torch.cuda.amp.autocast(dtype=self.model_args.compute_dtype): # support bf16 + logits, _, values = model(**input_kwargs) + + if values.size(0) != input_ids.size(0): # adapt to chatglm2 + values = torch.transpose(values, 0, 1) + + logprobs = logprobs_from_logits(logits[:, :-1, :], input_ids[:, 1:]) + masks = torch.zeros_like(attention_mask) + masks[:, :-1] = attention_mask[:, 1:] + + for j in range(len(query_batch)): + start = len(query_batch[j]) - 1 + if attention_mask[j, 0] == 0: # offset left padding + start += attention_mask[j, :].nonzero()[0].item() + end = start + len(response_batch[j]) + + if response_masks is not None: + response_masks_batch = torch.cat( + (torch.zeros_like(query_batch[j]), response_masks_batch[j]) + )[1:] + + masks[j, :start] = 0 + masks[j, end:] = 0 + if response_masks is not None: + masks[j, start:end] = masks[j, start:end] * response_masks_batch[j][start:end] + + if return_logits: + all_logits.append(logits) + else: + del logits + + all_values.append(values) + all_logprobs.append(logprobs) + all_masks.append(masks) + + return ( + torch.cat(all_logprobs), + torch.cat(all_logits)[:, :-1] if return_logits else None, + torch.cat(all_values)[:, :-1], + torch.cat(all_masks)[:, :-1], + ) + + def save_model(self, output_dir: Optional[str] = None) -> None: + r""" + Saves model checkpoint. + + Subclass and override to inject custom behavior. + """ + if self.args.should_save: + self._save(output_dir) diff --git a/llm_rl/src/llmtuner/tuner/ppo/utils.py b/llm_rl/src/llmtuner/tuner/ppo/utils.py new file mode 100644 index 00000000..74453a39 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/ppo/utils.py @@ -0,0 +1,35 @@ +import torch +from typing import TYPE_CHECKING, Dict, Literal, Optional + +if TYPE_CHECKING: + from transformers import PreTrainedModel + from trl import AutoModelForCausalLMWithValueHead + + +def replace_model(model: "AutoModelForCausalLMWithValueHead", target: Literal["default", "reward"]) -> None: + if target == "reward": # save default head temporarily + valuehead_state_dict: Dict[str, torch.Tensor] = model.v_head.state_dict() + setattr(model, "default_head_weight", valuehead_state_dict["summary.weight"].detach().clone()) + setattr(model, "default_head_bias", valuehead_state_dict["summary.bias"].detach().clone()) + + model.pretrained_model.set_adapter(target) # set the LoRA adapter to be active + model.v_head.load_state_dict({ + "summary.weight": model.get_buffer("{}_head_weight".format(target)).detach().clone(), + "summary.bias": model.get_buffer("{}_head_bias".format(target)).detach().clone() + }) + + +def dump_layernorm(model: "PreTrainedModel") -> Dict[str, torch.Tensor]: + layer_norm_params = {} + for name, param in model.named_parameters(): + if param.data.dtype == torch.float32: + layer_norm_params[name] = param.data.detach().clone() + param.data = param.data.to(model.config.torch_dtype) + + return layer_norm_params + + +def restore_layernorm(model: "PreTrainedModel", layernorm_params: Optional[Dict[str, torch.Tensor]] = None) -> None: + for name, param in model.named_parameters(): + if name in layernorm_params: + param.data = layernorm_params[name] diff --git a/llm_rl/src/llmtuner/tuner/ppo/workflow.py b/llm_rl/src/llmtuner/tuner/ppo/workflow.py new file mode 100644 index 00000000..4c35f628 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/ppo/workflow.py @@ -0,0 +1,92 @@ +# Inspired by: https://github.com/lvwerra/trl/blob/main/examples/research_projects/stack_llama/scripts/rl_training.py + +import math +from trl import PPOConfig +from torch.optim import AdamW +from typing import TYPE_CHECKING, Optional, List +from transformers import DataCollatorWithPadding +from transformers.optimization import get_scheduler + +from llmtuner.dsets import get_dataset, preprocess_dataset +from llmtuner.extras.callbacks import SavePeftModelCallback +from llmtuner.extras.ploting import plot_loss +from llmtuner.tuner.core import load_model_and_tokenizer +from llmtuner.tuner.ppo.trainer import CustomPPOTrainer + +if TYPE_CHECKING: + from transformers import Seq2SeqTrainingArguments, TrainerCallback + from llmtuner.hparams import ModelArguments, DataArguments, FinetuningArguments, GeneratingArguments + + +def run_ppo( + model_args: "ModelArguments", + data_args: "DataArguments", + training_args: "Seq2SeqTrainingArguments", + finetuning_args: "FinetuningArguments", + generating_args: "GeneratingArguments", + callbacks: Optional[List["TrainerCallback"]] = None +): + dataset = get_dataset(model_args, data_args) + model, tokenizer = load_model_and_tokenizer(model_args, finetuning_args, training_args.do_train, stage="ppo") + dataset = preprocess_dataset(dataset, tokenizer, data_args, training_args, stage="ppo") + + tokenizer.padding_side = "left" # use left-padding in generation while using right-padding in training + data_collator = DataCollatorWithPadding(tokenizer=tokenizer) + + ppo_config = PPOConfig( + model_name=model_args.model_name_or_path, + learning_rate=training_args.learning_rate, + mini_batch_size=training_args.per_device_train_batch_size, + batch_size=training_args.per_device_train_batch_size * training_args.gradient_accumulation_steps, + gradient_accumulation_steps=training_args.gradient_accumulation_steps, + ppo_epochs=1, + max_grad_norm=training_args.max_grad_norm, + seed=training_args.seed, + optimize_cuda_cache=True, + target=finetuning_args.ppo_target, + log_with=finetuning_args.ppo_logger, + use_score_scaling=finetuning_args.ppo_score_norm, + use_score_norm=finetuning_args.ppo_score_norm, + accelerator_kwargs={"step_scheduler_with_optimizer": False} + ) + + optimizer = AdamW(filter(lambda p: p.requires_grad, model.parameters()), lr=training_args.learning_rate) + if training_args.max_steps > 0: + num_training_steps = training_args.max_steps + else: + total_train_batch_size = ( + training_args.per_device_train_batch_size * training_args.gradient_accumulation_steps * training_args.world_size + ) + num_training_steps = training_args.num_train_epochs * math.ceil(len(dataset) / total_train_batch_size) + + lr_scheduler = get_scheduler( + training_args.lr_scheduler_type, + optimizer=optimizer, + num_warmup_steps=training_args.get_warmup_steps(num_training_steps), + num_training_steps=num_training_steps + ) + + # Initialize our Trainer + ppo_trainer = CustomPPOTrainer( + model_args=model_args, + training_args=training_args, + finetuning_args=finetuning_args, + generating_args=generating_args, + callbacks=callbacks + [SavePeftModelCallback()], + config=ppo_config, + model=model, + ref_model=None, + tokenizer=tokenizer, + dataset=dataset, + data_collator=data_collator, + optimizer=optimizer, + lr_scheduler=lr_scheduler + ) + + # Training + if training_args.do_train: + ppo_trainer.ppo_train() + ppo_trainer.save_model() + ppo_trainer.save_state() # must be called after save_model to have a folder + if ppo_trainer.is_world_process_zero() and model_args.plot_loss: + plot_loss(training_args.output_dir, keys=["loss", "reward"]) diff --git a/llm_rl/src/llmtuner/tuner/pt/__init__.py b/llm_rl/src/llmtuner/tuner/pt/__init__.py new file mode 100644 index 00000000..8ce509db --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/pt/__init__.py @@ -0,0 +1 @@ +from llmtuner.tuner.pt.workflow import run_pt diff --git a/llm_rl/src/llmtuner/tuner/pt/workflow.py b/llm_rl/src/llmtuner/tuner/pt/workflow.py new file mode 100644 index 00000000..66d08de7 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/pt/workflow.py @@ -0,0 +1,58 @@ +# Inspired by: https://github.com/huggingface/transformers/blob/v4.29.2/examples/pytorch/language-modeling/run_clm.py + +import math +from typing import TYPE_CHECKING, Optional, List +from transformers import DataCollatorForLanguageModeling, Trainer + +from llmtuner.dsets import get_dataset, preprocess_dataset, split_dataset +from llmtuner.extras.ploting import plot_loss +from llmtuner.tuner.core import load_model_and_tokenizer + +if TYPE_CHECKING: + from transformers import Seq2SeqTrainingArguments, TrainerCallback + from llmtuner.hparams import ModelArguments, DataArguments, FinetuningArguments + + +def run_pt( + model_args: "ModelArguments", + data_args: "DataArguments", + training_args: "Seq2SeqTrainingArguments", + finetuning_args: "FinetuningArguments", + callbacks: Optional[List["TrainerCallback"]] = None +): + dataset = get_dataset(model_args, data_args) + model, tokenizer = load_model_and_tokenizer(model_args, finetuning_args, training_args.do_train, stage="pt") + dataset = preprocess_dataset(dataset, tokenizer, data_args, training_args, stage="pt") + data_collator = DataCollatorForLanguageModeling(tokenizer=tokenizer, mlm=False) + + # Initialize our Trainer + trainer = Trainer( + model=model, + args=training_args, + tokenizer=tokenizer, + data_collator=data_collator, + callbacks=callbacks, + **split_dataset(dataset, data_args, training_args) + ) + + # Training + if training_args.do_train: + train_result = trainer.train(resume_from_checkpoint=training_args.resume_from_checkpoint) + trainer.log_metrics("train", train_result.metrics) + trainer.save_metrics("train", train_result.metrics) + trainer.save_state() + trainer.save_model() + if trainer.is_world_process_zero() and model_args.plot_loss: + plot_loss(training_args.output_dir, keys=["loss", "eval_loss"]) + + # Evaluation + if training_args.do_eval: + metrics = trainer.evaluate(metric_key_prefix="eval") + try: + perplexity = math.exp(metrics["eval_loss"]) + except OverflowError: + perplexity = float("inf") + + metrics["perplexity"] = perplexity + trainer.log_metrics("eval", metrics) + trainer.save_metrics("eval", metrics) diff --git a/llm_rl/src/llmtuner/tuner/rm/__init__.py b/llm_rl/src/llmtuner/tuner/rm/__init__.py new file mode 100644 index 00000000..54d3d943 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/rm/__init__.py @@ -0,0 +1 @@ +from llmtuner.tuner.rm.workflow import run_rm diff --git a/llm_rl/src/llmtuner/tuner/rm/collator.py b/llm_rl/src/llmtuner/tuner/rm/collator.py new file mode 100644 index 00000000..161f003d --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/rm/collator.py @@ -0,0 +1,27 @@ +import torch +from dataclasses import dataclass +from typing import Any, Dict, Sequence +from transformers import DataCollatorWithPadding + + +@dataclass +class PairwiseDataCollatorWithPadding(DataCollatorWithPadding): + r""" + Data collator for pairwise data. + """ + + def __call__(self, features: Sequence[Dict[str, Any]]) -> Dict[str, torch.Tensor]: + r""" + Pads batched data to the longest sequence in the batch. + + We generate 2 * n examples where the first n examples represent chosen examples and + the last n examples represent rejected examples. + """ + features = [ + { + "input_ids": feature["prompt_ids"] + feature[key], + "attention_mask": [1] * (len(feature["prompt_ids"]) + len(feature[key])) + } + for key in ("chosen_ids", "rejected_ids") for feature in features + ] + return super().__call__(features) diff --git a/llm_rl/src/llmtuner/tuner/rm/metric.py b/llm_rl/src/llmtuner/tuner/rm/metric.py new file mode 100644 index 00000000..db9c9243 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/rm/metric.py @@ -0,0 +1,7 @@ +import numpy as np +from typing import Dict, Sequence, Tuple, Union + + +def compute_accuracy(eval_preds: Sequence[Union[np.ndarray, Tuple[np.ndarray]]]) -> Dict[str, float]: + preds, _ = eval_preds + return {"accuracy": (preds[0] > preds[1]).sum() / len(preds[0])} diff --git a/llm_rl/src/llmtuner/tuner/rm/trainer.py b/llm_rl/src/llmtuner/tuner/rm/trainer.py new file mode 100644 index 00000000..80502937 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/rm/trainer.py @@ -0,0 +1,105 @@ +import os +import json +import torch +from typing import TYPE_CHECKING, Dict, List, Optional, Tuple, Union +from transformers import Trainer + +from llmtuner.extras.logging import get_logger + +if TYPE_CHECKING: + from transformers.trainer import PredictionOutput + from transformers.modeling_utils import PreTrainedModel + + +logger = get_logger(__name__) + + +class PairwiseTrainer(Trainer): + r""" + Inherits PeftTrainer to compute pairwise loss. + """ + + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + self.can_return_loss = True # override property to return eval_loss + + def compute_loss( + self, + model: "PreTrainedModel", + inputs: Dict[str, torch.Tensor], + return_outputs: Optional[bool] = False + ) -> Union[torch.Tensor, Tuple[torch.Tensor, List[torch.Tensor]]]: + r""" + Computes pairwise loss. The first n examples are chosen and the last n examples are rejected. + + Subclass and override to inject custom behavior. + + Note that the first element will be removed from the output tuple. + See: https://github.com/huggingface/transformers/blob/v4.30.2/src/transformers/trainer.py#L3509 + """ + # Compute rewards + _, _, values = model(**inputs, output_hidden_states=True, return_dict=True) + if values.size(0) != inputs["input_ids"].size(0): # adapt to chatglm2 + values = torch.transpose(values, 0, 1) + + # Split the inputs and rewards into two parts, chosen and rejected + batch_size = inputs["input_ids"].size(0) // 2 + chosen_input_ids, rejected_input_ids = inputs["input_ids"][:batch_size], inputs["input_ids"][batch_size:] + chosen_attn_mask, rejected_attn_mask = ( + inputs["attention_mask"][:batch_size], inputs["attention_mask"][batch_size:] + ) + chosen_rewards, rejected_rewards = values[:batch_size], values[batch_size:] + chosen_scores, rejected_scores = [], [] + + # Compute pairwise loss. Only backprop on the different tokens before padding + # Inspired by: https://github.com/CarperAI/trlx/blob/main/examples/summarize_rlhf/reward_model/reward_model.py + loss = 0 + for i in range(batch_size): + chosen_length = chosen_attn_mask[i].nonzero()[-1] + 1 + rejected_length = rejected_attn_mask[i].nonzero()[-1] + 1 + check_divergence = (chosen_input_ids[i] != rejected_input_ids[i]).nonzero() + + if len(check_divergence) == 0: + end_index = chosen_length + div_index = end_index - 1 + else: + end_index = max(chosen_length, rejected_length) + div_index = check_divergence[0] + + assert div_index > 0 + chosen_trunc_rewards = chosen_rewards[i, div_index:end_index] + rejected_trunc_rewards = rejected_rewards[i, div_index:end_index] + if return_outputs: # use the score on the EOS token for inference + chosen_scores.append(chosen_rewards[i, chosen_length-1]) + rejected_scores.append(rejected_rewards[i, rejected_length-1]) + loss += -torch.nn.functional.logsigmoid(chosen_trunc_rewards - rejected_trunc_rewards).mean() + + loss = loss / batch_size + if return_outputs: + chosen_scores, rejected_scores = torch.stack(chosen_scores), torch.stack(rejected_scores) + return loss, [loss, chosen_scores, rejected_scores] + + return loss + + def save_predictions( + self, + predict_results: "PredictionOutput" + ) -> None: + r""" + Saves model predictions to `output_dir`. + + A custom behavior that not contained in Seq2SeqTrainer. + """ + if not self.is_world_process_zero(): + return + + output_prediction_file = os.path.join(self.args.output_dir, "generated_predictions.jsonl") + logger.info(f"Saving prediction results to {output_prediction_file}") + + chosen_scores, rejected_scores = predict_results.predictions + + with open(output_prediction_file, "w", encoding="utf-8") as writer: + res: List[str] = [] + for c_score, r_score in zip(chosen_scores, rejected_scores): + res.append(json.dumps({"chosen": round(float(c_score), 2), "rejected": round(float(r_score), 2)})) + writer.write("\n".join(res)) diff --git a/llm_rl/src/llmtuner/tuner/rm/workflow.py b/llm_rl/src/llmtuner/tuner/rm/workflow.py new file mode 100644 index 00000000..6d2c4422 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/rm/workflow.py @@ -0,0 +1,68 @@ +# Inspired by: +# https://github.com/CarperAI/trlx/blob/main/examples/summarize_rlhf/reward_model/train_reward_model_gptj.py + +from typing import TYPE_CHECKING, Optional, List +from transformers import Seq2SeqTrainingArguments + +from llmtuner.dsets import get_dataset, preprocess_dataset, split_dataset +from llmtuner.extras.callbacks import SavePeftModelCallback +from llmtuner.extras.ploting import plot_loss +from llmtuner.tuner.core import load_model_and_tokenizer +from llmtuner.tuner.rm.metric import compute_accuracy +from llmtuner.tuner.rm.collator import PairwiseDataCollatorWithPadding +from llmtuner.tuner.rm.trainer import PairwiseTrainer + +if TYPE_CHECKING: + from transformers import TrainerCallback + from llmtuner.hparams import ModelArguments, DataArguments, FinetuningArguments + + +def run_rm( + model_args: "ModelArguments", + data_args: "DataArguments", + training_args: "Seq2SeqTrainingArguments", + finetuning_args: "FinetuningArguments", + callbacks: Optional[List["TrainerCallback"]] = None +): + dataset = get_dataset(model_args, data_args) + model, tokenizer = load_model_and_tokenizer(model_args, finetuning_args, training_args.do_train, stage="rm") + dataset = preprocess_dataset(dataset, tokenizer, data_args, training_args, stage="rm") + data_collator = PairwiseDataCollatorWithPadding(tokenizer, pad_to_multiple_of=4) + + training_args_dict = training_args.to_dict() + training_args_dict.update(dict(remove_unused_columns=False)) # important for pairwise dataset + training_args = Seq2SeqTrainingArguments(**training_args_dict) + + # Initialize our Trainer + trainer = PairwiseTrainer( + model=model, + args=training_args, + tokenizer=tokenizer, + data_collator=data_collator, + callbacks=callbacks + [SavePeftModelCallback()], + compute_metrics=compute_accuracy, + **split_dataset(dataset, data_args, training_args) + ) + + # Training + if training_args.do_train: + train_result = trainer.train() + trainer.log_metrics("train", train_result.metrics) + trainer.save_metrics("train", train_result.metrics) + trainer.save_state() + trainer.save_model() + if trainer.is_world_process_zero() and model_args.plot_loss: + plot_loss(training_args.output_dir, keys=["loss", "eval_loss"]) + + # Evaluation + if training_args.do_eval: + metrics = trainer.evaluate(metric_key_prefix="eval") + trainer.log_metrics("eval", metrics) + trainer.save_metrics("eval", metrics) + + # Predict + if training_args.do_predict: + predict_results = trainer.predict(dataset, metric_key_prefix="predict") + trainer.log_metrics("predict", predict_results.metrics) + trainer.save_metrics("predict", predict_results.metrics) + trainer.save_predictions(predict_results) diff --git a/llm_rl/src/llmtuner/tuner/sft/__init__.py b/llm_rl/src/llmtuner/tuner/sft/__init__.py new file mode 100644 index 00000000..493dd1a7 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/sft/__init__.py @@ -0,0 +1 @@ +from llmtuner.tuner.sft.workflow import run_sft diff --git a/llm_rl/src/llmtuner/tuner/sft/metric.py b/llm_rl/src/llmtuner/tuner/sft/metric.py new file mode 100644 index 00000000..812896ee --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/sft/metric.py @@ -0,0 +1,53 @@ +import numpy as np +from dataclasses import dataclass +from typing import TYPE_CHECKING, Dict, Sequence, Tuple, Union + +import jieba +from rouge_chinese import Rouge +from nltk.translate.bleu_score import sentence_bleu, SmoothingFunction + +from llmtuner.extras.constants import IGNORE_INDEX + +if TYPE_CHECKING: + from transformers.tokenization_utils import PreTrainedTokenizer + + +@dataclass +class ComputeMetrics: + r""" + Wraps the tokenizer into metric functions, used in Seq2SeqPeftTrainer. + """ + + tokenizer: "PreTrainedTokenizer" + + def __call__(self, eval_preds: Sequence[Union[np.ndarray, Tuple[np.ndarray]]]) -> Dict[str, float]: + r""" + Uses the model predictions to compute metrics. + """ + preds, labels = eval_preds + score_dict = {"rouge-1": [], "rouge-2": [], "rouge-l": [], "bleu-4": []} + + preds = np.where(preds != IGNORE_INDEX, preds, self.tokenizer.pad_token_id) + labels = np.where(labels != IGNORE_INDEX, labels, self.tokenizer.pad_token_id) + + decoded_preds = self.tokenizer.batch_decode(preds, skip_special_tokens=True) + decoded_labels = self.tokenizer.batch_decode(labels, skip_special_tokens=True) + + for pred, label in zip(decoded_preds, decoded_labels): + hypothesis = list(jieba.cut(pred)) + reference = list(jieba.cut(label)) + + if len(" ".join(hypothesis).split()) == 0 or len(" ".join(reference).split()) == 0: + result = {"rouge-1": {"f": 0.0}, "rouge-2": {"f": 0.0}, "rouge-l": {"f": 0.0}} + else: + rouge = Rouge() + scores = rouge.get_scores(" ".join(hypothesis), " ".join(reference)) + result = scores[0] + + for k, v in result.items(): + score_dict[k].append(round(v["f"] * 100, 4)) + + bleu_score = sentence_bleu([list(label)], list(pred), smoothing_function=SmoothingFunction().method3) + score_dict["bleu-4"].append(round(bleu_score * 100, 4)) + + return {k: float(np.mean(v)) for k, v in score_dict.items()} diff --git a/llm_rl/src/llmtuner/tuner/sft/trainer.py b/llm_rl/src/llmtuner/tuner/sft/trainer.py new file mode 100644 index 00000000..c65cd255 --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/sft/trainer.py @@ -0,0 +1,92 @@ +import os +import json +import torch +import numpy as np +import torch.nn as nn +from typing import TYPE_CHECKING, Any, Dict, List, Optional, Tuple, Union +from transformers import Seq2SeqTrainer + +from llmtuner.extras.constants import IGNORE_INDEX +from llmtuner.extras.logging import get_logger + +if TYPE_CHECKING: + from transformers.trainer import PredictionOutput + + +logger = get_logger(__name__) + + +class CustomSeq2SeqTrainer(Seq2SeqTrainer): + r""" + Inherits PeftTrainer to compute generative metrics such as BLEU and ROUGE. + """ + + def prediction_step( + self, + model: nn.Module, + inputs: Dict[str, Union[torch.Tensor, Any]], + prediction_loss_only: bool, + ignore_keys: Optional[List[str]] = None, + ) -> Tuple[Optional[float], Optional[torch.Tensor], Optional[torch.Tensor]]: + r""" + Removes the prompt part in the generated tokens. + + Subclass and override to inject custom behavior. + """ + labels = inputs["labels"].detach().clone() if "labels" in inputs else None # backup labels + if self.args.predict_with_generate: + assert self.tokenizer.padding_side == "left", "This method only accepts left-padded tensor." + prompt_len, label_len = inputs["input_ids"].size(-1), inputs["labels"].size(-1) + if prompt_len > label_len: + inputs["labels"] = self._pad_tensors_to_target_len(inputs["labels"], inputs["input_ids"]) + if label_len > prompt_len: + inputs["labels"] = inputs["labels"][:, :prompt_len] # truncate the labels instead of padding the inputs + + loss, generated_tokens, _ = super().prediction_step( + model, inputs, prediction_loss_only=prediction_loss_only, ignore_keys=ignore_keys + ) + if generated_tokens is not None and self.args.predict_with_generate: + generated_tokens[:, :prompt_len] = self.tokenizer.pad_token_id + generated_tokens = generated_tokens.contiguous() + + return loss, generated_tokens, labels + + def _pad_tensors_to_target_len( + self, + src_tensor: torch.Tensor, + tgt_tensor: torch.Tensor + ) -> torch.Tensor: + r""" + Pads the tensor to the same length as the target tensor. + """ + assert self.tokenizer.pad_token_id is not None, "Pad token is required." + padded_tensor = self.tokenizer.pad_token_id * torch.ones_like(tgt_tensor) + padded_tensor[:, -src_tensor.shape[-1]:] = src_tensor # adopt left-padding + return padded_tensor.contiguous() # in contiguous memory + + def save_predictions( + self, + predict_results: "PredictionOutput" + ) -> None: + r""" + Saves model predictions to `output_dir`. + + A custom behavior that not contained in Seq2SeqTrainer. + """ + if not self.is_world_process_zero(): + return + + output_prediction_file = os.path.join(self.args.output_dir, "generated_predictions.jsonl") + logger.info(f"Saving prediction results to {output_prediction_file}") + + preds = np.where(predict_results.predictions != IGNORE_INDEX, predict_results.predictions, self.tokenizer.pad_token_id) + labels = np.where(predict_results.label_ids != IGNORE_INDEX, predict_results.label_ids, self.tokenizer.pad_token_id) + + decoded_preds = self.tokenizer.batch_decode(preds, skip_special_tokens=True, clean_up_tokenization_spaces=True) + decoded_labels = self.tokenizer.batch_decode(labels, skip_special_tokens=True, clean_up_tokenization_spaces=True) + + with open(output_prediction_file, "w", encoding="utf-8") as writer: + res: List[str] = [] + for pred, label in zip(decoded_preds, decoded_labels): + res.append(json.dumps({"label": label, "predict": pred}, ensure_ascii=False)) + writer.write("\n".join(res)) diff --git a/llm_rl/src/llmtuner/tuner/sft/workflow.py b/llm_rl/src/llmtuner/tuner/sft/workflow.py new file mode 100644 index 00000000..8d53605d --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/sft/workflow.py @@ -0,0 +1,90 @@ +# Inspired by: https://github.com/huggingface/transformers/blob/v4.29.2/examples/pytorch/summarization/run_summarization.py + +from typing import TYPE_CHECKING, Optional, List +from transformers import DataCollatorForSeq2Seq, Seq2SeqTrainingArguments + +from llmtuner.dsets import get_dataset, preprocess_dataset, split_dataset +from llmtuner.extras.constants import IGNORE_INDEX +from llmtuner.extras.misc import get_logits_processor +from llmtuner.extras.ploting import plot_loss +from llmtuner.tuner.core import load_model_and_tokenizer +from llmtuner.tuner.sft.metric import ComputeMetrics +from llmtuner.tuner.sft.trainer import CustomSeq2SeqTrainer + +if TYPE_CHECKING: + from transformers import TrainerCallback + from llmtuner.hparams import ModelArguments, DataArguments, FinetuningArguments, GeneratingArguments + + +def run_sft( + model_args: "ModelArguments", + data_args: "DataArguments", + training_args: "Seq2SeqTrainingArguments", + finetuning_args: "FinetuningArguments", + generating_args: "GeneratingArguments", + callbacks: Optional[List["TrainerCallback"]] = None +): + dataset = get_dataset(model_args, data_args) + model, tokenizer = load_model_and_tokenizer(model_args, finetuning_args, training_args.do_train, stage="sft") + dataset = preprocess_dataset(dataset, tokenizer, data_args, training_args, stage="sft") + + if training_args.predict_with_generate: + tokenizer.padding_side = "left" # use left-padding in generation + + data_collator = DataCollatorForSeq2Seq( + tokenizer=tokenizer, + pad_to_multiple_of=4 if tokenizer.padding_side == "right" else None, # for shift short attention + label_pad_token_id=IGNORE_INDEX if data_args.ignore_pad_token_for_loss else tokenizer.pad_token_id + ) + + # Override the decoding parameters of Seq2SeqTrainer + training_args_dict = training_args.to_dict() + training_args_dict.update(dict( + generation_max_length=training_args.generation_max_length or data_args.cutoff_len, + generation_num_beams=data_args.eval_num_beams or training_args.generation_num_beams + )) + training_args = Seq2SeqTrainingArguments(**training_args_dict) + + # Initialize our Trainer + trainer = CustomSeq2SeqTrainer( + model=model, + args=training_args, + tokenizer=tokenizer, + data_collator=data_collator, + callbacks=callbacks, + compute_metrics=ComputeMetrics(tokenizer) if training_args.predict_with_generate else None, + **split_dataset(dataset, data_args, training_args) + ) + + # Keyword arguments for `model.generate` + gen_kwargs = generating_args.to_dict() + gen_kwargs["eos_token_id"] = [tokenizer.eos_token_id] + tokenizer.additional_special_tokens_ids + gen_kwargs["pad_token_id"] = tokenizer.pad_token_id + gen_kwargs["logits_processor"] = get_logits_processor() + + # Training + if training_args.do_train: + train_result = trainer.train(resume_from_checkpoint=training_args.resume_from_checkpoint) + trainer.log_metrics("train", train_result.metrics) + trainer.save_metrics("train", train_result.metrics) + trainer.save_state() + trainer.save_model() + if trainer.is_world_process_zero() and model_args.plot_loss: + plot_loss(training_args.output_dir, keys=["loss", "eval_loss"]) + + # Evaluation + if training_args.do_eval: + metrics = trainer.evaluate(metric_key_prefix="eval", **gen_kwargs) + if training_args.predict_with_generate: # eval_loss will be wrong if predict_with_generate is enabled + metrics.pop("eval_loss", None) + trainer.log_metrics("eval", metrics) + trainer.save_metrics("eval", metrics) + + # Predict + if training_args.do_predict: + predict_results = trainer.predict(dataset, metric_key_prefix="predict", **gen_kwargs) + if training_args.predict_with_generate: # predict_loss will be wrong if predict_with_generate is enabled + predict_results.metrics.pop("predict_loss", None) + trainer.log_metrics("predict", predict_results.metrics) + trainer.save_metrics("predict", predict_results.metrics) + trainer.save_predictions(predict_results) diff --git a/llm_rl/src/llmtuner/tuner/tune.py b/llm_rl/src/llmtuner/tuner/tune.py new file mode 100644 index 00000000..4eb7f78f --- /dev/null +++ b/llm_rl/src/llmtuner/tuner/tune.py @@ -0,0 +1,51 @@ +from typing import TYPE_CHECKING, Any, Dict, List, Optional + +from llmtuner.extras.callbacks import LogCallback +from llmtuner.extras.logging import get_logger +from llmtuner.tuner.core import get_train_args, get_infer_args, load_model_and_tokenizer +from llmtuner.tuner.pt import run_pt +from llmtuner.tuner.sft import run_sft +from llmtuner.tuner.rm import run_rm +from llmtuner.tuner.ppo import run_ppo +from llmtuner.tuner.dpo import run_dpo + +if TYPE_CHECKING: + from transformers import TrainerCallback + + +logger = get_logger(__name__) + + +def run_exp(args: Optional[Dict[str, Any]] = None, callbacks: Optional[List["TrainerCallback"]] = None): + model_args, data_args, training_args, finetuning_args, generating_args = get_train_args(args) + callbacks = [LogCallback()] if callbacks is None else callbacks + + if finetuning_args.stage == "pt": + run_pt(model_args, data_args, training_args, finetuning_args, callbacks) + elif finetuning_args.stage == "sft": + run_sft(model_args, data_args, training_args, finetuning_args, generating_args, callbacks) + elif finetuning_args.stage == "rm": + run_rm(model_args, data_args, training_args, finetuning_args, callbacks) + elif finetuning_args.stage == "ppo": + run_ppo(model_args, data_args, training_args, finetuning_args, generating_args, callbacks) + elif finetuning_args.stage == "dpo": + run_dpo(model_args, data_args, training_args, finetuning_args, callbacks) + else: + raise ValueError("Unknown task.") + + +def export_model(args: Optional[Dict[str, Any]] = None, max_shard_size: Optional[str] = "10GB"): + model_args, _, finetuning_args, _ = get_infer_args(args) + model, tokenizer = load_model_and_tokenizer(model_args, finetuning_args) + model.config.use_cache = True + model.save_pretrained(model_args.export_dir, max_shard_size=max_shard_size) + try: + tokenizer.padding_side = "left" # restore padding side + tokenizer.init_kwargs["padding_side"] = "left" + tokenizer.save_pretrained(model_args.export_dir) + except: + logger.warning("Cannot save tokenizer, please copy the files manually.") + + +if __name__ == "__main__": + run_exp() diff --git a/llm_rl/src/llmtuner/webui/__init__.py b/llm_rl/src/llmtuner/webui/__init__.py new file mode 100644 index 00000000..a27c7f6e --- /dev/null +++ b/llm_rl/src/llmtuner/webui/__init__.py @@ -0,0 +1 @@ +from llmtuner.webui.interface import create_ui, create_web_demo diff --git a/llm_rl/src/llmtuner/webui/chatter.py b/llm_rl/src/llmtuner/webui/chatter.py new file mode 100644 index 00000000..57eadb01 --- /dev/null +++ b/llm_rl/src/llmtuner/webui/chatter.py @@ -0,0 +1,101 @@ +import gradio as gr +from gradio.components import Component # cannot use TYPE_CHECKING here +from typing import TYPE_CHECKING, Any, Dict, Generator, List, Optional, Tuple + +from llmtuner.chat.stream_chat import ChatModel +from llmtuner.extras.misc import torch_gc +from llmtuner.hparams import GeneratingArguments +from llmtuner.webui.common import get_save_dir +from llmtuner.webui.locales import ALERTS + +if TYPE_CHECKING: + from llmtuner.webui.manager import Manager + + +class WebChatModel(ChatModel): + + def __init__(self, manager: "Manager", lazy_init: Optional[bool] = True) -> None: + self.manager = manager + self.model = None + self.tokenizer = None + self.generating_args = GeneratingArguments() + if not lazy_init: + super().__init__() + + @property + def loaded(self) -> bool: + return self.model is not None + + def load_model(self, data: Dict[Component, Any]) -> Generator[str, None, None]: + get = lambda name: data[self.manager.get_elem_by_name(name)] + lang = get("top.lang") + error = "" + if self.loaded: + error = ALERTS["err_exists"][lang] + elif not get("top.model_name"): + error = ALERTS["err_no_model"][lang] + elif not get("top.model_path"): + error = ALERTS["err_no_path"][lang] + + if error: + gr.Warning(error) + yield error + return + + if get("top.checkpoints"): + checkpoint_dir = ",".join([ + get_save_dir(get("top.model_name"), get("top.finetuning_type"), ckpt) for ckpt in get("top.checkpoints") + ]) + else: + checkpoint_dir = None + + yield ALERTS["info_loading"][lang] + args = dict( + model_name_or_path=get("top.model_path"), + checkpoint_dir=checkpoint_dir, + finetuning_type=get("top.finetuning_type"), + quantization_bit=int(get("top.quantization_bit")) if get("top.quantization_bit") in ["8", "4"] else None, + template=get("top.template"), + system_prompt=get("top.system_prompt"), + flash_attn=get("top.flash_attn"), + shift_attn=get("top.shift_attn"), + rope_scaling=get("top.rope_scaling") if get("top.rope_scaling") in ["linear", "dynamic"] else None + ) + super().__init__(args) + + yield ALERTS["info_loaded"][lang] + + def unload_model(self, data: Dict[Component, Any]) -> Generator[str, None, None]: + lang = data[self.manager.get_elem_by_name("top.lang")] + yield ALERTS["info_unloading"][lang] + self.model = None + self.tokenizer = None + torch_gc() + yield ALERTS["info_unloaded"][lang] + + def predict( + self, + chatbot: List[Tuple[str, str]], + query: str, + history: List[Tuple[str, str]], + system: str, + max_new_tokens: int, + top_p: float, + temperature: float + ) -> Generator[Tuple[List[Tuple[str, str]], List[Tuple[str, str]]], None, None]: + chatbot.append([query, ""]) + response = "" + for new_text in self.stream_chat( + query, history, system, max_new_tokens=max_new_tokens, top_p=top_p, temperature=temperature + ): + response += new_text + new_history = history + [(query, response)] + chatbot[-1] = [query, self.postprocess(response)] + yield chatbot, new_history + + def postprocess(self, response: str) -> str: + blocks = response.split("```") + for i, block in enumerate(blocks): + if i % 2 == 0: + blocks[i] = block.replace("<", "<").replace(">", ">") + return "```".join(blocks) diff --git a/llm_rl/src/llmtuner/webui/common.py b/llm_rl/src/llmtuner/webui/common.py new file mode 100644 index 00000000..5a6c16d3 --- /dev/null +++ b/llm_rl/src/llmtuner/webui/common.py @@ -0,0 +1,103 @@ +import os +import json +import gradio as gr +from typing import Any, Dict, Optional +from transformers.utils import ( + WEIGHTS_NAME, + WEIGHTS_INDEX_NAME, + SAFE_WEIGHTS_NAME, + SAFE_WEIGHTS_INDEX_NAME, + ADAPTER_WEIGHTS_NAME, + ADAPTER_SAFE_WEIGHTS_NAME +) + +from llmtuner.extras.constants import DEFAULT_MODULE, DEFAULT_TEMPLATE, SUPPORTED_MODELS, TRAINING_STAGES + + +DEFAULT_CACHE_DIR = "cache" +DEFAULT_DATA_DIR = "data" +DEFAULT_SAVE_DIR = "saves" +USER_CONFIG = "user.config" +DATA_CONFIG = "dataset_info.json" +CKPT_NAMES = [ + WEIGHTS_NAME, + WEIGHTS_INDEX_NAME, + SAFE_WEIGHTS_NAME, + SAFE_WEIGHTS_INDEX_NAME, + ADAPTER_WEIGHTS_NAME, + ADAPTER_SAFE_WEIGHTS_NAME +] + + +def get_save_dir(*args) -> os.PathLike: + return os.path.join(DEFAULT_SAVE_DIR, *args) + + +def get_config_path() -> os.PathLike: + return os.path.join(DEFAULT_CACHE_DIR, USER_CONFIG) + + +def load_config() -> Dict[str, Any]: + try: + with open(get_config_path(), "r", encoding="utf-8") as f: + return json.load(f) + except: + return {"lang": None, "last_model": None, "path_dict": {}, "cache_dir": None} + + +def save_config(lang: str, model_name: Optional[str] = None, model_path: Optional[str] = None) -> None: + os.makedirs(DEFAULT_CACHE_DIR, exist_ok=True) + user_config = load_config() + user_config["lang"] = lang or user_config["lang"] + if model_name: + user_config["last_model"] = model_name + user_config["path_dict"][model_name] = model_path + with open(get_config_path(), "w", encoding="utf-8") as f: + json.dump(user_config, f, indent=2, ensure_ascii=False) + + +def get_model_path(model_name: str) -> str: + user_config = load_config() + return user_config["path_dict"].get(model_name, None) or SUPPORTED_MODELS.get(model_name, "") + + +def get_module(model_name: str) -> str: + return DEFAULT_MODULE.get(model_name.split("-")[0], "q_proj,v_proj") + + +def get_template(model_name: str) -> str: + if model_name.endswith("Chat") and model_name.split("-")[0] in DEFAULT_TEMPLATE: + return DEFAULT_TEMPLATE[model_name.split("-")[0]] + return "default" + + +def list_checkpoint(model_name: str, finetuning_type: str) -> Dict[str, Any]: + checkpoints = [] + if model_name: + save_dir = get_save_dir(model_name, finetuning_type) + if save_dir and os.path.isdir(save_dir): + for checkpoint in os.listdir(save_dir): + if ( + os.path.isdir(os.path.join(save_dir, checkpoint)) + and any([os.path.isfile(os.path.join(save_dir, checkpoint, name)) for name in CKPT_NAMES]) + ): + checkpoints.append(checkpoint) + return gr.update(value=[], choices=checkpoints) + + +def load_dataset_info(dataset_dir: str) -> Dict[str, Any]: + try: + with open(os.path.join(dataset_dir, DATA_CONFIG), "r", encoding="utf-8") as f: + return json.load(f) + except: + print("Cannot find {} in {}.".format(DATA_CONFIG, dataset_dir)) + return {} + + +def list_dataset( + dataset_dir: Optional[str] = None, training_stage: Optional[str] = list(TRAINING_STAGES.keys())[0] +) -> Dict[str, Any]: + dataset_info = load_dataset_info(dataset_dir if dataset_dir is not None else DEFAULT_DATA_DIR) + ranking = TRAINING_STAGES[training_stage] in ["rm", "dpo"] + datasets = [k for k, v in dataset_info.items() if v.get("ranking", False) == ranking] + return gr.update(value=[], choices=datasets) diff --git a/llm_rl/src/llmtuner/webui/components/__init__.py b/llm_rl/src/llmtuner/webui/components/__init__.py new file mode 100644 index 00000000..32228b8e --- /dev/null +++ b/llm_rl/src/llmtuner/webui/components/__init__.py @@ -0,0 +1,6 @@ +from llmtuner.webui.components.top import create_top +from llmtuner.webui.components.train import create_train_tab +from llmtuner.webui.components.eval import create_eval_tab +from llmtuner.webui.components.infer import create_infer_tab +from llmtuner.webui.components.export import create_export_tab +from llmtuner.webui.components.chatbot import create_chat_box diff --git a/llm_rl/src/llmtuner/webui/components/chatbot.py b/llm_rl/src/llmtuner/webui/components/chatbot.py new file mode 100644 index 00000000..13e2dd4d --- /dev/null +++ b/llm_rl/src/llmtuner/webui/components/chatbot.py @@ -0,0 +1,49 @@ +import gradio as gr +from typing import TYPE_CHECKING, Dict, Optional, Tuple + +if TYPE_CHECKING: + from gradio.blocks import Block + from gradio.components import Component + from llmtuner.webui.engine import Engine + + +def create_chat_box( + engine: "Engine", + visible: Optional[bool] = False +) -> Tuple["Block", "Component", "Component", Dict[str, "Component"]]: + with gr.Box(visible=visible) as chat_box: + chatbot = gr.Chatbot() + history = gr.State([]) + with gr.Row(): + with gr.Column(scale=4): + system = gr.Textbox(show_label=False) + query = gr.Textbox(show_label=False, lines=8) + submit_btn = gr.Button(variant="primary") + + with gr.Column(scale=1): + clear_btn = gr.Button() + gen_kwargs = engine.chatter.generating_args + max_new_tokens = gr.Slider(10, 2048, value=gen_kwargs.max_new_tokens, step=1) + top_p = gr.Slider(0.01, 1, value=gen_kwargs.top_p, step=0.01) + temperature = gr.Slider(0.01, 1.5, value=gen_kwargs.temperature, step=0.01) + + submit_btn.click( + engine.chatter.predict, + [chatbot, query, history, system, max_new_tokens, top_p, temperature], + [chatbot, history], + show_progress=True + ).then( + lambda: gr.update(value=""), outputs=[query] + ) + + clear_btn.click(lambda: ([], []), outputs=[chatbot, history], show_progress=True) + + return chat_box, chatbot, history, dict( + system=system, + query=query, + submit_btn=submit_btn, + clear_btn=clear_btn, + max_new_tokens=max_new_tokens, + top_p=top_p, + temperature=temperature + ) diff --git a/llm_rl/src/llmtuner/webui/components/data.py b/llm_rl/src/llmtuner/webui/components/data.py new file mode 100644 index 00000000..effa39da --- /dev/null +++ b/llm_rl/src/llmtuner/webui/components/data.py @@ -0,0 +1,103 @@ +import os +import json +import gradio as gr +from typing import TYPE_CHECKING, Any, Dict, Tuple + +from llmtuner.webui.common import DATA_CONFIG + +if TYPE_CHECKING: + from gradio.components import Component + + +PAGE_SIZE = 2 + + +def prev_page(page_index: int) -> int: + return page_index - 1 if page_index > 0 else page_index + + +def next_page(page_index: int, total_num: int) -> int: + return page_index + 1 if (page_index + 1) * PAGE_SIZE < total_num else page_index + + +def can_preview(dataset_dir: str, dataset: list) -> Dict[str, Any]: + with open(os.path.join(dataset_dir, DATA_CONFIG), "r", encoding="utf-8") as f: + dataset_info = json.load(f) + + if ( + len(dataset) > 0 + and "file_name" in dataset_info[dataset[0]] + and os.path.isfile(os.path.join(dataset_dir, dataset_info[dataset[0]]["file_name"])) + ): + return gr.update(interactive=True) + else: + return gr.update(interactive=False) + + +def get_preview(dataset_dir: str, dataset: list, page_index: int) -> Tuple[int, list, Dict[str, Any]]: + with open(os.path.join(dataset_dir, DATA_CONFIG), "r", encoding="utf-8") as f: + dataset_info = json.load(f) + + data_file: str = dataset_info[dataset[0]]["file_name"] + with open(os.path.join(dataset_dir, data_file), "r", encoding="utf-8") as f: + if data_file.endswith(".json"): + data = json.load(f) + elif data_file.endswith(".jsonl"): + data = [json.loads(line) for line in f] + else: + data = [line for line in f] + return len(data), data[PAGE_SIZE * page_index : PAGE_SIZE * (page_index + 1)], gr.update(visible=True) + + +def create_preview_box(dataset_dir: "gr.Textbox", dataset: "gr.Dropdown") -> Dict[str, "Component"]: + data_preview_btn = gr.Button(interactive=False, scale=1) + with gr.Column(visible=False, elem_classes="modal-box") as preview_box: + with gr.Row(): + preview_count = gr.Number(value=0, interactive=False, precision=0) + page_index = gr.Number(value=0, interactive=False, precision=0) + + with gr.Row(): + prev_btn = gr.Button() + next_btn = gr.Button() + close_btn = gr.Button() + + with gr.Row(): + preview_samples = gr.JSON(interactive=False) + + dataset.change( + can_preview, [dataset_dir, dataset], [data_preview_btn], queue=False + ).then( + lambda: 0, outputs=[page_index], queue=False + ) + data_preview_btn.click( + get_preview, + [dataset_dir, dataset, page_index], + [preview_count, preview_samples, preview_box], + queue=False + ) + prev_btn.click( + prev_page, [page_index], [page_index], queue=False + ).then( + get_preview, + [dataset_dir, dataset, page_index], + [preview_count, preview_samples, preview_box], + queue=False + ) + next_btn.click( + next_page, [page_index, preview_count], [page_index], queue=False + ).then( + get_preview, + [dataset_dir, dataset, page_index], + [preview_count, preview_samples, preview_box], + queue=False + ) + close_btn.click(lambda: gr.update(visible=False), outputs=[preview_box], queue=False) + return dict( + data_preview_btn=data_preview_btn, + preview_count=preview_count, + page_index=page_index, + prev_btn=prev_btn, + next_btn=next_btn, + close_btn=close_btn, + preview_samples=preview_samples + ) diff --git a/llm_rl/src/llmtuner/webui/components/eval.py b/llm_rl/src/llmtuner/webui/components/eval.py new file mode 100644 index 00000000..36c994a6 --- /dev/null +++ b/llm_rl/src/llmtuner/webui/components/eval.py @@ -0,0 +1,70 @@ +import gradio as gr +from typing import TYPE_CHECKING, Dict + +from llmtuner.webui.common import list_dataset, DEFAULT_DATA_DIR +from llmtuner.webui.components.data import create_preview_box + +if TYPE_CHECKING: + from gradio.components import Component + from llmtuner.webui.engine import Engine + + +def create_eval_tab(engine: "Engine") -> Dict[str, "Component"]: + input_elems = engine.manager.get_base_elems() + elem_dict = dict() + + with gr.Row(): + dataset_dir = gr.Textbox(value=DEFAULT_DATA_DIR, scale=2) + dataset = gr.Dropdown(multiselect=True, scale=4) + preview_elems = create_preview_box(dataset_dir, dataset) + + dataset_dir.change(list_dataset, [dataset_dir], [dataset], queue=False) + + input_elems.update({dataset_dir, dataset}) + elem_dict.update(dict(dataset_dir=dataset_dir, dataset=dataset, **preview_elems)) + + with gr.Row(): + cutoff_len = gr.Slider(value=1024, minimum=4, maximum=8192, step=1) + max_samples = gr.Textbox(value="100000") + batch_size = gr.Slider(value=8, minimum=1, maximum=512, step=1) + predict = gr.Checkbox(value=True) + + input_elems.update({cutoff_len, max_samples, batch_size, predict}) + elem_dict.update(dict( + cutoff_len=cutoff_len, max_samples=max_samples, batch_size=batch_size, predict=predict + )) + + with gr.Row(): + max_new_tokens = gr.Slider(10, 2048, value=128, step=1) + top_p = gr.Slider(0.01, 1, value=0.7, step=0.01) + temperature = gr.Slider(0.01, 1.5, value=0.95, step=0.01) + + input_elems.update({max_new_tokens, top_p, temperature}) + elem_dict.update(dict( + max_new_tokens=max_new_tokens, top_p=top_p, temperature=temperature + )) + + with gr.Row(): + cmd_preview_btn = gr.Button() + start_btn = gr.Button() + stop_btn = gr.Button() + + with gr.Row(): + resume_btn = gr.Checkbox(visible=False, interactive=False, value=False) + process_bar = gr.Slider(visible=False, interactive=False) + + with gr.Box(): + output_box = gr.Markdown() + + output_elems = [output_box, process_bar] + elem_dict.update(dict( + cmd_preview_btn=cmd_preview_btn, start_btn=start_btn, stop_btn=stop_btn, + resume_btn=resume_btn, process_bar=process_bar, output_box=output_box + )) + + cmd_preview_btn.click(engine.runner.preview_eval, input_elems, output_elems) + start_btn.click(engine.runner.run_eval, input_elems, output_elems) + stop_btn.click(engine.runner.set_abort, queue=False) + resume_btn.change(engine.runner.monitor, outputs=output_elems) + + return elem_dict diff --git a/llm_rl/src/llmtuner/webui/components/export.py b/llm_rl/src/llmtuner/webui/components/export.py new file mode 100644 index 00000000..d16fa3d1 --- /dev/null +++ b/llm_rl/src/llmtuner/webui/components/export.py @@ -0,0 +1,79 @@ +import gradio as gr +from typing import TYPE_CHECKING, Dict, Generator, List + +from llmtuner.tuner import export_model +from llmtuner.webui.common import get_save_dir +from llmtuner.webui.locales import ALERTS + +if TYPE_CHECKING: + from gradio.components import Component + from llmtuner.webui.engine import Engine + + +def save_model( + lang: str, + model_name: str, + model_path: str, + checkpoints: List[str], + finetuning_type: str, + template: str, + max_shard_size: int, + export_dir: str +) -> Generator[str, None, None]: + error = "" + if not model_name: + error = ALERTS["err_no_model"][lang] + elif not model_path: + error = ALERTS["err_no_path"][lang] + elif not checkpoints: + error = ALERTS["err_no_checkpoint"][lang] + elif not export_dir: + error = ALERTS["err_no_export_dir"][lang] + + if error: + gr.Warning(error) + yield error + return + + args = dict( + model_name_or_path=model_path, + checkpoint_dir=",".join([get_save_dir(model_name, finetuning_type, ckpt) for ckpt in checkpoints]), + finetuning_type=finetuning_type, + template=template, + export_dir=export_dir + ) + + yield ALERTS["info_exporting"][lang] + export_model(args, max_shard_size="{}GB".format(max_shard_size)) + yield ALERTS["info_exported"][lang] + + +def create_export_tab(engine: "Engine") -> Dict[str, "Component"]: + with gr.Row(): + export_dir = gr.Textbox() + max_shard_size = gr.Slider(value=10, minimum=1, maximum=100) + + export_btn = gr.Button() + info_box = gr.Textbox(show_label=False, interactive=False) + + export_btn.click( + save_model, + [ + engine.manager.get_elem_by_name("top.lang"), + engine.manager.get_elem_by_name("top.model_name"), + engine.manager.get_elem_by_name("top.model_path"), + engine.manager.get_elem_by_name("top.checkpoints"), + engine.manager.get_elem_by_name("top.finetuning_type"), + engine.manager.get_elem_by_name("top.template"), + max_shard_size, + export_dir + ], + [info_box] + ) + + return dict( + export_dir=export_dir, + max_shard_size=max_shard_size, + export_btn=export_btn, + info_box=info_box + ) diff --git a/llm_rl/src/llmtuner/webui/components/infer.py b/llm_rl/src/llmtuner/webui/components/infer.py new file mode 100644 index 00000000..d6dd7eed --- /dev/null +++ b/llm_rl/src/llmtuner/webui/components/infer.py @@ -0,0 +1,39 @@ +import gradio as gr +from typing import TYPE_CHECKING, Dict + +from llmtuner.webui.components.chatbot import create_chat_box + +if TYPE_CHECKING: + from gradio.components import Component + from llmtuner.webui.engine import Engine + + +def create_infer_tab(engine: "Engine") -> Dict[str, "Component"]: + input_elems = engine.manager.get_base_elems() + elem_dict = dict() + + with gr.Row(): + load_btn = gr.Button() + unload_btn = gr.Button() + + info_box = gr.Textbox(show_label=False, interactive=False) + elem_dict.update(dict(load_btn=load_btn, unload_btn=unload_btn, info_box=info_box)) + + chat_box, chatbot, history, chat_elems = create_chat_box(engine, visible=False) + elem_dict.update(dict(chat_box=chat_box, **chat_elems)) + + load_btn.click( + engine.chatter.load_model, input_elems, [info_box] + ).then( + lambda: gr.update(visible=engine.chatter.loaded), outputs=[chat_box] + ) + + unload_btn.click( + engine.chatter.unload_model, input_elems, [info_box] + ).then( + lambda: ([], []), outputs=[chatbot, history] + ).then( + lambda: gr.update(visible=engine.chatter.loaded), outputs=[chat_box] + ) + + return elem_dict diff --git a/llm_rl/src/llmtuner/webui/components/top.py b/llm_rl/src/llmtuner/webui/components/top.py new file mode 100644 index 00000000..c6299cab --- /dev/null +++ b/llm_rl/src/llmtuner/webui/components/top.py @@ -0,0 +1,74 @@ +import gradio as gr +from typing import TYPE_CHECKING, Dict + +from llmtuner.extras.constants import METHODS, SUPPORTED_MODELS +from llmtuner.extras.template import templates +from llmtuner.webui.common import get_model_path, get_template, list_checkpoint, save_config +from llmtuner.webui.utils import can_quantize + +if TYPE_CHECKING: + from gradio.components import Component + + +def create_top() -> Dict[str, "Component"]: + available_models = list(SUPPORTED_MODELS.keys()) + ["Custom"] + + with gr.Row(): + lang = gr.Dropdown(choices=["en", "zh"], scale=1) + model_name = gr.Dropdown(choices=available_models, scale=3) + model_path = gr.Textbox(scale=3) + + with gr.Row(): + finetuning_type = gr.Dropdown(choices=METHODS, value="lora", scale=1) + checkpoints = gr.Dropdown(multiselect=True, scale=5) + refresh_btn = gr.Button(scale=1) + + with gr.Accordion(label="Advanced config", open=False) as advanced_tab: + with gr.Row(): + quantization_bit = gr.Dropdown(choices=["none", "8", "4"], value="none", scale=1) + template = gr.Dropdown(choices=list(templates.keys()), value="default", scale=1) + system_prompt = gr.Textbox(scale=2) + + with gr.Accordion(label="Model config (LLaMA only)", open=False) as llama_tab: + with gr.Row(): + with gr.Column(): + flash_attn = gr.Checkbox(value=False) + shift_attn = gr.Checkbox(value=False) + rope_scaling = gr.Radio(choices=["none", "linear", "dynamic"], value="none") + + model_name.change( + list_checkpoint, [model_name, finetuning_type], [checkpoints], queue=False + ).then( + get_model_path, [model_name], [model_path], queue=False + ).then( + get_template, [model_name], [template], queue=False + ) # do not save config since the below line will save + + model_path.change(save_config, inputs=[lang, model_name, model_path], queue=False) + + finetuning_type.change( + list_checkpoint, [model_name, finetuning_type], [checkpoints], queue=False + ).then( + can_quantize, [finetuning_type], [quantization_bit], queue=False + ) + + refresh_btn.click( + list_checkpoint, [model_name, finetuning_type], [checkpoints], queue=False + ) + + return dict( + lang=lang, + model_name=model_name, + model_path=model_path, + finetuning_type=finetuning_type, + checkpoints=checkpoints, + refresh_btn=refresh_btn, + advanced_tab=advanced_tab, + quantization_bit=quantization_bit, + template=template, + system_prompt=system_prompt, + llama_tab=llama_tab, + flash_attn=flash_attn, + shift_attn=shift_attn, + rope_scaling=rope_scaling + ) diff --git a/llm_rl/src/llmtuner/webui/components/train.py b/llm_rl/src/llmtuner/webui/components/train.py new file mode 100644 index 00000000..11109c97 --- /dev/null +++ b/llm_rl/src/llmtuner/webui/components/train.py @@ -0,0 +1,154 @@ +import gradio as gr +from typing import TYPE_CHECKING, Dict +from transformers.trainer_utils import SchedulerType + +from llmtuner.extras.constants import TRAINING_STAGES +from llmtuner.webui.common import list_checkpoint, list_dataset, DEFAULT_DATA_DIR +from llmtuner.webui.components.data import create_preview_box +from llmtuner.webui.utils import gen_plot + +if TYPE_CHECKING: + from gradio.components import Component + from llmtuner.webui.engine import Engine + + +def create_train_tab(engine: "Engine") -> Dict[str, "Component"]: + input_elems = engine.manager.get_base_elems() + elem_dict = dict() + + with gr.Row(): + training_stage = gr.Dropdown( + choices=list(TRAINING_STAGES.keys()), value=list(TRAINING_STAGES.keys())[0], scale=2 + ) + dataset_dir = gr.Textbox(value=DEFAULT_DATA_DIR, scale=2) + dataset = gr.Dropdown(multiselect=True, scale=4) + preview_elems = create_preview_box(dataset_dir, dataset) + + training_stage.change(list_dataset, [dataset_dir, training_stage], [dataset], queue=False) + dataset_dir.change(list_dataset, [dataset_dir, training_stage], [dataset], queue=False) + + input_elems.update({training_stage, dataset_dir, dataset}) + elem_dict.update(dict( + training_stage=training_stage, dataset_dir=dataset_dir, dataset=dataset, **preview_elems + )) + + with gr.Row(): + cutoff_len = gr.Slider(value=1024, minimum=4, maximum=8192, step=1) + learning_rate = gr.Textbox(value="5e-5") + num_train_epochs = gr.Textbox(value="3.0") + max_samples = gr.Textbox(value="100000") + compute_type = gr.Radio(choices=["fp16", "bf16"], value="fp16") + + input_elems.update({cutoff_len, learning_rate, num_train_epochs, max_samples, compute_type}) + elem_dict.update(dict( + cutoff_len=cutoff_len, learning_rate=learning_rate, num_train_epochs=num_train_epochs, + max_samples=max_samples, compute_type=compute_type + )) + + with gr.Row(): + batch_size = gr.Slider(value=4, minimum=1, maximum=512, step=1) + gradient_accumulation_steps = gr.Slider(value=4, minimum=1, maximum=512, step=1) + lr_scheduler_type = gr.Dropdown( + choices=[scheduler.value for scheduler in SchedulerType], value="cosine" + ) + max_grad_norm = gr.Textbox(value="1.0") + val_size = gr.Slider(value=0, minimum=0, maximum=1, step=0.001) + + input_elems.update({batch_size, gradient_accumulation_steps, lr_scheduler_type, max_grad_norm, val_size}) + elem_dict.update(dict( + batch_size=batch_size, gradient_accumulation_steps=gradient_accumulation_steps, + lr_scheduler_type=lr_scheduler_type, max_grad_norm=max_grad_norm, val_size=val_size + )) + + with gr.Accordion(label="Advanced config", open=False) as advanced_tab: + with gr.Row(): + logging_steps = gr.Slider(value=5, minimum=5, maximum=1000, step=5) + save_steps = gr.Slider(value=100, minimum=10, maximum=5000, step=10) + warmup_steps = gr.Slider(value=0, minimum=0, maximum=5000, step=1) + neft_alpha = gr.Slider(value=0, minimum=0, maximum=10, step=0.1) + + with gr.Column(): + train_on_prompt = gr.Checkbox(value=False) + upcast_layernorm = gr.Checkbox(value=False) + + input_elems.update({logging_steps, save_steps, warmup_steps, neft_alpha, train_on_prompt, upcast_layernorm}) + elem_dict.update(dict( + advanced_tab=advanced_tab, logging_steps=logging_steps, save_steps=save_steps, warmup_steps=warmup_steps, + neft_alpha=neft_alpha, train_on_prompt=train_on_prompt, upcast_layernorm=upcast_layernorm + )) + + with gr.Accordion(label="LoRA config", open=False) as lora_tab: + with gr.Row(): + lora_rank = gr.Slider(value=8, minimum=1, maximum=1024, step=1, scale=1) + lora_dropout = gr.Slider(value=0.1, minimum=0, maximum=1, step=0.01, scale=1) + lora_target = gr.Textbox(scale=1) + additional_target = gr.Textbox(scale=1) + resume_lora_training = gr.Checkbox(value=True, scale=1) + + input_elems.update({lora_rank, lora_dropout, lora_target, additional_target, resume_lora_training}) + elem_dict.update(dict( + lora_tab=lora_tab, lora_rank=lora_rank, lora_dropout=lora_dropout, lora_target=lora_target, + additional_target=additional_target, resume_lora_training=resume_lora_training, + )) + + with gr.Accordion(label="RLHF config", open=False) as rlhf_tab: + with gr.Row(): + dpo_beta = gr.Slider(value=0.1, minimum=0, maximum=1, step=0.01, scale=1) + reward_model = gr.Dropdown(scale=3) + refresh_btn = gr.Button(scale=1) + + refresh_btn.click( + list_checkpoint, + [engine.manager.get_elem_by_name("top.model_name"), engine.manager.get_elem_by_name("top.finetuning_type")], + [reward_model], + queue=False + ) + + input_elems.update({dpo_beta, reward_model}) + elem_dict.update(dict(rlhf_tab=rlhf_tab, dpo_beta=dpo_beta, reward_model=reward_model, refresh_btn=refresh_btn)) + + with gr.Row(): + cmd_preview_btn = gr.Button() + start_btn = gr.Button() + stop_btn = gr.Button() + + with gr.Row(): + with gr.Column(scale=3): + with gr.Row(): + output_dir = gr.Textbox() + + with gr.Row(): + resume_btn = gr.Checkbox(visible=False, interactive=False, value=False) + process_bar = gr.Slider(visible=False, interactive=False) + + with gr.Box(): + output_box = gr.Markdown() + + with gr.Column(scale=1): + loss_viewer = gr.Plot() + + input_elems.add(output_dir) + output_elems = [output_box, process_bar] + + cmd_preview_btn.click(engine.runner.preview_train, input_elems, output_elems) + start_btn.click(engine.runner.run_train, input_elems, output_elems) + stop_btn.click(engine.runner.set_abort, queue=False) + resume_btn.change(engine.runner.monitor, outputs=output_elems) + + elem_dict.update(dict( + cmd_preview_btn=cmd_preview_btn, start_btn=start_btn, stop_btn=stop_btn, output_dir=output_dir, + resume_btn=resume_btn, process_bar=process_bar, output_box=output_box, loss_viewer=loss_viewer + )) + + output_box.change( + gen_plot, + [ + engine.manager.get_elem_by_name("top.model_name"), + engine.manager.get_elem_by_name("top.finetuning_type"), + output_dir + ], + loss_viewer, + queue=False + ) + + return elem_dict diff --git a/llm_rl/src/llmtuner/webui/css.py b/llm_rl/src/llmtuner/webui/css.py new file mode 100644 index 00000000..c86fb96b --- /dev/null +++ b/llm_rl/src/llmtuner/webui/css.py @@ -0,0 +1,20 @@ +CSS = r""" +.modal-box { + position: fixed !important; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); /* center horizontally */ + max-width: 1000px; + max-height: 750px; + overflow-y: auto; + background-color: var(--input-background-fill); + flex-wrap: nowrap !important; + border: 2px solid black !important; + z-index: 1000; + padding: 10px; +} + +.dark .modal-box { + border: 2px solid white !important; +} +""" diff --git a/llm_rl/src/llmtuner/webui/engine.py b/llm_rl/src/llmtuner/webui/engine.py new file mode 100644 index 00000000..661dfb48 --- /dev/null +++ b/llm_rl/src/llmtuner/webui/engine.py @@ -0,0 +1,57 @@ +import gradio as gr +from gradio.components import Component # cannot use TYPE_CHECKING here +from typing import Any, Dict, Generator, Optional + +from llmtuner.webui.chatter import WebChatModel +from llmtuner.webui.common import get_model_path, list_dataset, load_config +from llmtuner.webui.locales import LOCALES +from llmtuner.webui.manager import Manager +from llmtuner.webui.runner import Runner +from llmtuner.webui.utils import get_time + + +class Engine: + + def __init__(self, pure_chat: Optional[bool] = False) -> None: + self.pure_chat = pure_chat + self.manager: "Manager" = Manager() + self.runner: "Runner" = Runner(self.manager) + self.chatter: "WebChatModel" = WebChatModel(manager=self.manager, lazy_init=(not pure_chat)) + + def _form_dict(self, resume_dict: Dict[str, Dict[str, Any]]): + return {self.manager.get_elem_by_name(k): gr.update(**v) for k, v in resume_dict.items()} + + def resume(self) -> Generator[Dict[Component, Dict[str, Any]], None, None]: + user_config = load_config() + lang = user_config.get("lang", None) or "en" + + init_dict = { + "top.lang": {"value": lang}, + "infer.chat_box": {"visible": self.chatter.loaded} + } + + if not self.pure_chat: + init_dict["train.dataset"] = {"choices": list_dataset()["choices"]} + init_dict["eval.dataset"] = {"choices": list_dataset()["choices"]} + + if user_config.get("last_model", None): + init_dict["top.model_name"] = {"value": user_config["last_model"]} + init_dict["top.model_path"] = {"value": get_model_path(user_config["last_model"])} + + yield self._form_dict(init_dict) + + if not self.pure_chat: + if self.runner.alive: + yield {elem: gr.update(value=value) for elem, value in self.runner.running_data.items()} + if self.runner.do_train: + yield self._form_dict({"train.resume_btn": {"value": True}}) + else: + yield self._form_dict({"eval.resume_btn": {"value": True}}) + else: + yield self._form_dict({"train.output_dir": {"value": get_time()}}) + + def change_lang(self, lang: str) -> Dict[Component, Dict[str, Any]]: + return { + component: gr.update(**LOCALES[name][lang]) + for elems in self.manager.all_elems.values() for name, component in elems.items() if name in LOCALES + } diff --git a/llm_rl/src/llmtuner/webui/interface.py b/llm_rl/src/llmtuner/webui/interface.py new file mode 100644 index 00000000..ba663f24 --- /dev/null +++ b/llm_rl/src/llmtuner/webui/interface.py @@ -0,0 +1,66 @@ +import gradio as gr +from transformers.utils.versions import require_version + +from llmtuner.webui.components import ( + create_top, + create_train_tab, + create_eval_tab, + create_infer_tab, + create_export_tab, + create_chat_box +) +from llmtuner.webui.common import save_config +from llmtuner.webui.css import CSS +from llmtuner.webui.engine import Engine + + +require_version("gradio>=3.38.0,<4.0.0", "To fix: pip install \"gradio>=3.38.0,<4.0.0\"") + + +def create_ui() -> gr.Blocks: + engine = Engine(pure_chat=False) + + with gr.Blocks(title="LLaMA Board", css=CSS) as demo: + engine.manager.all_elems["top"] = create_top() + lang: "gr.Dropdown" = engine.manager.get_elem_by_name("top.lang") + + with gr.Tab("Train"): + engine.manager.all_elems["train"] = create_train_tab(engine) + + with gr.Tab("Evaluate"): + engine.manager.all_elems["eval"] = create_eval_tab(engine) + + with gr.Tab("Chat"): + engine.manager.all_elems["infer"] = create_infer_tab(engine) + + with gr.Tab("Export"): + engine.manager.all_elems["export"] = create_export_tab(engine) + + demo.load(engine.resume, outputs=engine.manager.list_elems()) + lang.change(engine.change_lang, [lang], engine.manager.list_elems(), queue=False) + lang.input(save_config, inputs=[lang], queue=False) + + return demo + + +def create_web_demo() -> gr.Blocks: + engine = Engine(pure_chat=True) + + with gr.Blocks(title="Web Demo", css=CSS) as demo: + lang = gr.Dropdown(choices=["en", "zh"]) + engine.manager.all_elems["top"] = dict(lang=lang) + + chat_box, _, _, chat_elems = create_chat_box(engine, visible=True) + engine.manager.all_elems["infer"] = dict(chat_box=chat_box, **chat_elems) + + demo.load(engine.resume, outputs=engine.manager.list_elems()) + lang.change(engine.change_lang, [lang], engine.manager.list_elems(), queue=False) + lang.input(save_config, inputs=[lang], queue=False) + + return demo + + +if __name__ == "__main__": + demo = create_ui() + demo.queue() + demo.launch(server_name="0.0.0.0", server_port=7860, share=False, inbrowser=True) diff --git a/llm_rl/src/llmtuner/webui/locales.py b/llm_rl/src/llmtuner/webui/locales.py new file mode 100644 index 00000000..cc2a1842 --- /dev/null +++ b/llm_rl/src/llmtuner/webui/locales.py @@ -0,0 +1,698 @@ +LOCALES = { + "lang": { + "en": { + "label": "Lang" + }, + "zh": { + "label": "语言" + } + }, + "model_name": { + "en": { + "label": "Model name" + }, + "zh": { + "label": "模型名称" + } + }, + "model_path": { + "en": { + "label": "Model path", + "info": "Path to pretrained model or model identifier from Hugging Face." + }, + "zh": { + "label": "模型路径", + "info": "本地模型的文件路径或 Hugging Face 的模型标识符。" + } + }, + "finetuning_type": { + "en": { + "label": "Finetuning method" + }, + "zh": { + "label": "微调方法" + } + }, + "checkpoints": { + "en": { + "label": "Checkpoints" + }, + "zh": { + "label": "模型断点" + } + }, + "refresh_btn": { + "en": { + "value": "Refresh checkpoints" + }, + "zh": { + "value": "刷新断点" + } + }, + "advanced_tab": { + "en": { + "label": "Advanced configurations" + }, + "zh": { + "label": "高级设置" + } + }, + "quantization_bit": { + "en": { + "label": "Quantization bit", + "info": "Enable 4/8-bit model quantization (QLoRA)." + }, + "zh": { + "label": "量化等级", + "info": "启用 4/8 比特模型量化(QLoRA)。" + } + }, + "template": { + "en": { + "label": "Prompt template", + "info": "The template used in constructing prompts." + }, + "zh": { + "label": "提示模板", + "info": "构建提示词时使用的模板" + } + }, + "system_prompt": { + "en": { + "label": "System prompt (optional)", + "info": "A sequence used as the default system prompt." + }, + "zh": { + "label": "系统提示词(非必填)", + "info": "默认使用的系统提示词" + } + }, + "llama_tab": { + "en": { + "label": "Model configurations (LLaMA only)" + }, + "zh": { + "label": "模型设置(仅LLaMA)" + } + }, + "flash_attn": { + "en": { + "label": "Use FlashAttention-2" + }, + "zh": { + "label": "使用 FlashAttention-2" + } + }, + "shift_attn": { + "en": { + "label": "Use shift short attention (S^2-Attn)" + }, + "zh": { + "label": "使用 shift short attention (S^2-Attn)" + } + }, + "rope_scaling": { + "en": { + "label": "RoPE scaling" + }, + "zh": { + "label": "RoPE 插值方法" + } + }, + "training_stage": { + "en": { + "label": "Stage", + "info": "The stage to perform in training." + }, + "zh": { + "label": "训练阶段", + "info": "目前采用的训练方式。" + } + }, + "dataset_dir": { + "en": { + "label": "Data dir", + "info": "Path of the data directory." + }, + "zh": { + "label": "数据路径", + "info": "数据文件夹的路径。" + } + }, + "dataset": { + "en": { + "label": "Dataset" + }, + "zh": { + "label": "数据集" + } + }, + "data_preview_btn": { + "en": { + "value": "Preview dataset" + }, + "zh": { + "value": "预览数据集" + } + }, + "preview_count": { + "en": { + "label": "Count" + }, + "zh": { + "label": "数量" + } + }, + "page_index": { + "en": { + "label": "Page" + }, + "zh": { + "label": "页数" + } + }, + "prev_btn": { + "en": { + "value": "Prev" + }, + "zh": { + "value": "上一页" + } + }, + "next_btn": { + "en": { + "value": "Next" + }, + "zh": { + "value": "下一页" + } + }, + "close_btn": { + "en": { + "value": "Close" + }, + "zh": { + "value": "关闭" + } + }, + "preview_samples": { + "en": { + "label": "Samples" + }, + "zh": { + "label": "样例" + } + }, + "cutoff_len": { + "en": { + "label": "Cutoff length", + "info": "Max tokens in input sequence." + }, + "zh": { + "label": "截断长度", + "info": "输入序列分词后的最大长度。" + } + }, + "learning_rate": { + "en": { + "label": "Learning rate", + "info": "Initial learning rate for AdamW." + }, + "zh": { + "label": "学习率", + "info": "AdamW 优化器的初始学习率。" + } + }, + "num_train_epochs": { + "en": { + "label": "Epochs", + "info": "Total number of training epochs to perform." + }, + "zh": { + "label": "训练轮数", + "info": "需要执行的训练总轮数。" + } + }, + "max_samples": { + "en": { + "label": "Max samples", + "info": "Maximum samples per dataset." + }, + "zh": { + "label": "最大样本数", + "info": "每个数据集最多使用的样本数。" + } + }, + "compute_type": { + "en": { + "label": "Compute type", + "info": "Whether to use fp16 or bf16 mixed precision training." + }, + "zh": { + "label": "计算类型", + "info": "是否启用 FP16 或 BF16 混合精度训练。" + } + }, + "batch_size": { + "en": { + "label": "Batch size", + "info": "Number of samples to process per GPU." + }, + "zh":{ + "label": "批处理大小", + "info": "每块 GPU 上处理的样本数量。" + } + }, + "gradient_accumulation_steps": { + "en": { + "label": "Gradient accumulation", + "info": "Number of gradient accumulation steps." + }, + "zh": { + "label": "梯度累积", + "info": "梯度累积的步数。" + } + }, + "lr_scheduler_type": { + "en": { + "label": "LR Scheduler", + "info": "Name of learning rate scheduler.", + }, + "zh": { + "label": "学习率调节器", + "info": "采用的学习率调节器名称。" + } + }, + "max_grad_norm": { + "en": { + "label": "Maximum gradient norm", + "info": "Norm for gradient clipping.." + }, + "zh": { + "label": "最大梯度范数", + "info": "用于梯度裁剪的范数。" + } + }, + "val_size": { + "en": { + "label": "Val size", + "info": "Proportion of data in the dev set." + }, + "zh": { + "label": "验证集比例", + "info": "验证集占全部样本的百分比。" + } + }, + "logging_steps": { + "en": { + "label": "Logging steps", + "info": "Number of steps between two logs." + }, + "zh": { + "label": "日志间隔", + "info": "每两次日志输出间的更新步数。" + } + }, + "save_steps": { + "en": { + "label": "Save steps", + "info": "Number of steps between two checkpoints." + }, + "zh": { + "label": "保存间隔", + "info": "每两次断点保存间的更新步数。" + } + }, + "warmup_steps": { + "en": { + "label": "Warmup steps", + "info": "Number of steps used for warmup." + }, + "zh": { + "label": "预热步数", + "info": "学习率预热采用的步数。" + } + }, + "neft_alpha": { + "en": { + "label": "NEFTune Alpha", + "info": "Magnitude of noise adding to embedding vectors." + }, + "zh": { + "label": "NEFTune 噪声参数", + "info": "嵌入向量所添加的噪声大小。" + } + }, + "train_on_prompt": { + "en": { + "label": "Train on prompt", + "info": "Compute loss on the prompt tokens in supervised fine-tuning." + }, + "zh": { + "label": "计算输入损失", + "info": "在监督微调时候计算输入序列的损失。" + } + }, + "upcast_layernorm": { + "en": { + "label": "Upcast LayerNorm", + "info": "Upcast weights of layernorm in float32." + }, + "zh": { + "label": "缩放归一化层", + "info": "将归一化层权重缩放至 32 位浮点数。" + } + }, + "lora_tab": { + "en": { + "label": "LoRA configurations" + }, + "zh": { + "label": "LoRA 参数设置" + } + }, + "lora_rank": { + "en": { + "label": "LoRA rank", + "info": "The rank of LoRA matrices." + }, + "zh": { + "label": "LoRA 秩", + "info": "LoRA 矩阵的秩。" + } + }, + "lora_dropout": { + "en": { + "label": "LoRA Dropout", + "info": "Dropout ratio of LoRA weights." + }, + "zh": { + "label": "LoRA 随机丢弃", + "info": "LoRA 权重随机丢弃的概率。" + } + }, + "lora_target": { + "en": { + "label": "LoRA modules (optional)", + "info": "Name(s) of target modules to apply LoRA. Use commas to separate multiple modules." + }, + "zh": { + "label": "LoRA 作用模块(非必填)", + "info": "应用 LoRA 的目标模块名称。使用英文逗号分隔多个名称。" + } + }, + "additional_target": { + "en": { + "label": "Additional modules (optional)", + "info": "Name(s) of modules apart from LoRA layers to be set as trainable. Use commas to separate multiple modules." + }, + "zh": { + "label": "附加模块(非必填)", + "info": "除 LoRA 层以外的可训练模块名称。使用英文逗号分隔多个名称。" + } + }, + "resume_lora_training": { + "en": { + "label": "Resume LoRA training", + "info": "Whether to resume training from the last LoRA weights or create new lora weights." + }, + "zh": { + "label": "继续上次的训练", + "info": "接着上次的 LoRA 权重训练或创建一个新的 LoRA 权重。" + } + }, + "rlhf_tab": { + "en": { + "label": "RLHF configurations" + }, + "zh": { + "label": "RLHF 参数设置" + } + }, + "dpo_beta": { + "en": { + "label": "DPO beta", + "info": "Value of the beta parameter in the DPO loss." + }, + "zh": { + "label": "DPO beta 参数", + "info": "DPO 损失函数中 beta 超参数大小。" + } + }, + "reward_model": { + "en": { + "label": "Reward model", + "info": "Checkpoint of the reward model for PPO training. (Needs to refresh checkpoints)" + }, + "zh": { + "label": "奖励模型", + "info": "PPO 训练中奖励模型的断点路径。(需要刷新断点)" + } + }, + "cmd_preview_btn": { + "en": { + "value": "Preview command" + }, + "zh": { + "value": "预览命令" + } + }, + "start_btn": { + "en": { + "value": "Start" + }, + "zh": { + "value": "开始" + } + }, + "stop_btn": { + "en": { + "value": "Abort" + }, + "zh": { + "value": "中断" + } + }, + "output_dir": { + "en": { + "label": "Checkpoint name", + "info": "Directory to save checkpoint." + }, + "zh": { + "label": "断点名称", + "info": "保存模型断点的文件夹名称。" + } + }, + "output_box": { + "en": { + "value": "Ready." + }, + "zh": { + "value": "准备就绪。" + } + }, + "loss_viewer": { + "en": { + "label": "Loss" + }, + "zh": { + "label": "损失" + } + }, + "predict": { + "en": { + "label": "Save predictions" + }, + "zh": { + "label": "保存预测结果" + } + }, + "load_btn": { + "en": { + "value": "Load model" + }, + "zh": { + "value": "加载模型" + } + }, + "unload_btn": { + "en": { + "value": "Unload model" + }, + "zh": { + "value": "卸载模型" + } + }, + "info_box": { + "en": { + "value": "Model unloaded, please load a model first." + }, + "zh": { + "value": "模型未加载,请先加载模型。" + } + }, + "system": { + "en": { + "placeholder": "System prompt (optional)" + }, + "zh": { + "placeholder": "系统提示词(非必填)" + } + }, + "query": { + "en": { + "placeholder": "Input..." + }, + "zh": { + "placeholder": "输入..." + } + }, + "submit_btn": { + "en": { + "value": "Submit" + }, + "zh": { + "value": "提交" + } + }, + "clear_btn": { + "en": { + "value": "Clear history" + }, + "zh": { + "value": "清空历史" + } + }, + "max_length": { + "en": { + "label": "Maximum length" + }, + "zh": { + "label": "最大长度" + } + }, + "max_new_tokens": { + "en": { + "label": "Maximum new tokens" + }, + "zh": { + "label": "最大生成长度" + } + }, + "top_p": { + "en": { + "label": "Top-p" + }, + "zh": { + "label": "Top-p 采样值" + } + }, + "temperature": { + "en": { + "label": "Temperature" + }, + "zh": { + "label": "温度系数" + } + }, + "export_dir": { + "en": { + "label": "Export dir", + "info": "Directory to save exported model." + }, + "zh": { + "label": "导出目录", + "info": "保存导出模型的文件夹路径。" + } + }, + "max_shard_size": { + "en": { + "label": "Max shard size (GB)", + "info": "The maximum size for a model file." + }, + "zh": { + "label": "最大分块大小(GB)", + "info": "模型文件的最大大小。" + } + }, + "export_btn": { + "en": { + "value": "Export" + }, + "zh": { + "value": "开始导出" + } + } +} + + +ALERTS = { + "err_conflict": { + "en": "A process is in running, please abort it firstly.", + "zh": "任务已存在,请先中断训练。" + }, + "err_exists": { + "en": "You have loaded a model, please unload it first.", + "zh": "模型已存在,请先卸载模型。" + }, + "err_no_model": { + "en": "Please select a model.", + "zh": "请选择模型。" + }, + "err_no_path": { + "en": "Model not found.", + "zh": "模型未找到。" + }, + "err_no_dataset": { + "en": "Please choose a dataset.", + "zh": "请选择数据集。" + }, + "err_no_checkpoint": { + "en": "Please select a checkpoint.", + "zh": "请选择断点。" + }, + "err_no_export_dir": { + "en": "Please provide export dir.", + "zh": "请填写导出目录" + }, + "err_failed": { + "en": "Failed.", + "zh": "训练出错。" + }, + "info_aborting": { + "en": "Aborted, wait for terminating...", + "zh": "训练中断,正在等待线程结束……" + }, + "info_aborted": { + "en": "Ready.", + "zh": "准备就绪。" + }, + "info_finished": { + "en": "Finished.", + "zh": "训练完毕。" + }, + "info_loading": { + "en": "Loading model...", + "zh": "加载中……" + }, + "info_unloading": { + "en": "Unloading model...", + "zh": "卸载中……" + }, + "info_loaded": { + "en": "Model loaded, now you can chat with your model!", + "zh": "模型已加载,可以开始聊天了!" + }, + "info_unloaded": { + "en": "Model unloaded.", + "zh": "模型已卸载。" + }, + "info_exporting": { + "en": "Exporting model...", + "zh": "正在导出模型……" + }, + "info_exported": { + "en": "Model exported.", + "zh": "模型导出完成。" + } +} diff --git a/llm_rl/src/llmtuner/webui/manager.py b/llm_rl/src/llmtuner/webui/manager.py new file mode 100644 index 00000000..ca067aea --- /dev/null +++ b/llm_rl/src/llmtuner/webui/manager.py @@ -0,0 +1,35 @@ +from typing import TYPE_CHECKING, Dict, List, Set + +if TYPE_CHECKING: + from gradio.components import Component + + +class Manager: + + def __init__(self) -> None: + self.all_elems: Dict[str, Dict[str, "Component"]] = {} + + def get_elem_by_name(self, name: str) -> "Component": + r""" + Example: top.lang, train.dataset + """ + tab_name, elem_name = name.split(".") + return self.all_elems[tab_name][elem_name] + + def get_base_elems(self) -> Set["Component"]: + return { + self.all_elems["top"]["lang"], + self.all_elems["top"]["model_name"], + self.all_elems["top"]["model_path"], + self.all_elems["top"]["checkpoints"], + self.all_elems["top"]["finetuning_type"], + self.all_elems["top"]["quantization_bit"], + self.all_elems["top"]["template"], + self.all_elems["top"]["system_prompt"], + self.all_elems["top"]["flash_attn"], + self.all_elems["top"]["shift_attn"], + self.all_elems["top"]["rope_scaling"] + } + + def list_elems(self) -> List["Component"]: + return [elem for elems in self.all_elems.values() for elem in elems.values()] diff --git a/llm_rl/src/llmtuner/webui/runner.py b/llm_rl/src/llmtuner/webui/runner.py new file mode 100644 index 00000000..ab9e9ffc --- /dev/null +++ b/llm_rl/src/llmtuner/webui/runner.py @@ -0,0 +1,254 @@ +import os +import time +import logging +import gradio as gr +from threading import Thread +from gradio.components import Component # cannot use TYPE_CHECKING here +from typing import TYPE_CHECKING, Any, Dict, Generator, List, Tuple + +import transformers +from transformers.trainer import TRAINING_ARGS_NAME + +from llmtuner.extras.callbacks import LogCallback +from llmtuner.extras.constants import TRAINING_STAGES +from llmtuner.extras.logging import LoggerHandler +from llmtuner.extras.misc import torch_gc +from llmtuner.tuner import run_exp +from llmtuner.webui.common import get_module, get_save_dir, load_config +from llmtuner.webui.locales import ALERTS +from llmtuner.webui.utils import gen_cmd, get_eval_results, update_process_bar + +if TYPE_CHECKING: + from llmtuner.webui.manager import Manager + + +class Runner: + + def __init__(self, manager: "Manager") -> None: + self.manager = manager + """ Resume """ + self.thread: "Thread" = None + self.do_train = True + self.running_data: Dict["Component", Any] = None + self.monitor_inputs: Dict[str, str] = None + """ State """ + self.aborted = False + self.running = False + """ Handler """ + self.logger_handler = LoggerHandler() + self.logger_handler.setLevel(logging.INFO) + logging.root.addHandler(self.logger_handler) + transformers.logging.add_handler(self.logger_handler) + + @property + def alive(self) -> bool: + return self.thread is not None + + def set_abort(self) -> None: + self.aborted = True + self.running = False + + def _initialize(self, data: Dict[Component, Any], do_train: bool) -> str: + get = lambda name: data[self.manager.get_elem_by_name(name)] + lang, model_name, model_path = get("top.lang"), get("top.model_name"), get("top.model_path") + dataset = get("train.dataset") if do_train else get("eval.dataset") + + if self.running: + return ALERTS["err_conflict"][lang] + + if not model_name: + return ALERTS["err_no_model"][lang] + + if not model_path: + return ALERTS["err_no_path"][lang] + + if len(dataset) == 0: + return ALERTS["err_no_dataset"][lang] + + self.aborted = False + self.logger_handler.reset() + self.trainer_callback = LogCallback(self) + return "" + + def _finalize(self, lang: str, finish_info: str) -> str: + self.thread = None + self.running = False + torch_gc() + if self.aborted: + return ALERTS["info_aborted"][lang] + else: + return finish_info + + def _parse_train_args(self, data: Dict[Component, Any]) -> Dict[str, Any]: + get = lambda name: data[self.manager.get_elem_by_name(name)] + user_config = load_config() + + if get("top.checkpoints"): + checkpoint_dir = ",".join([ + get_save_dir(get("top.model_name"), get("top.finetuning_type"), ckpt) for ckpt in get("top.checkpoints") + ]) + else: + checkpoint_dir = None + + args = dict( + stage=TRAINING_STAGES[get("train.training_stage")], + model_name_or_path=get("top.model_path"), + do_train=True, + cache_dir=user_config.get("cache_dir", None), + checkpoint_dir=checkpoint_dir, + finetuning_type=get("top.finetuning_type"), + quantization_bit=int(get("top.quantization_bit")) if get("top.quantization_bit") in ["8", "4"] else None, + template=get("top.template"), + system_prompt=get("top.system_prompt"), + flash_attn=get("top.flash_attn"), + shift_attn=get("top.shift_attn"), + rope_scaling=get("top.rope_scaling") if get("top.rope_scaling") in ["linear", "dynamic"] else None, + dataset_dir=get("train.dataset_dir"), + dataset=",".join(get("train.dataset")), + cutoff_len=get("train.cutoff_len"), + learning_rate=float(get("train.learning_rate")), + num_train_epochs=float(get("train.num_train_epochs")), + max_samples=int(get("train.max_samples")), + per_device_train_batch_size=get("train.batch_size"), + gradient_accumulation_steps=get("train.gradient_accumulation_steps"), + lr_scheduler_type=get("train.lr_scheduler_type"), + max_grad_norm=float(get("train.max_grad_norm")), + logging_steps=get("train.logging_steps"), + save_steps=get("train.save_steps"), + warmup_steps=get("train.warmup_steps"), + neft_alpha=get("train.neft_alpha"), + train_on_prompt=get("train.train_on_prompt"), + upcast_layernorm=get("train.upcast_layernorm"), + lora_rank=get("train.lora_rank"), + lora_dropout=get("train.lora_dropout"), + lora_target=get("train.lora_target") or get_module(get("top.model_name")), + additional_target=get("train.additional_target") if get("train.additional_target") else None, + resume_lora_training=get("train.resume_lora_training"), + output_dir=get_save_dir(get("top.model_name"), get("top.finetuning_type"), get("train.output_dir")) + ) + args[get("train.compute_type")] = True + args["disable_tqdm"] = True + + if TRAINING_STAGES[get("train.training_stage")] in ["rm", "ppo", "dpo"]: + args["resume_lora_training"] = (args["quantization_bit"] is not None) + + if args["quantization_bit"] is not None: + args["upcast_layernorm"] = True + + if args["stage"] == "ppo": + args["reward_model"] = get("train.reward_model") + + if args["stage"] == "dpo": + args["dpo_beta"] = get("train.dpo_beta") + + if get("train.val_size") > 1e-6 and args["stage"] != "ppo": + args["val_size"] = get("train.val_size") + args["evaluation_strategy"] = "steps" + args["eval_steps"] = get("train.save_steps") + args["load_best_model_at_end"] = True + + return args + + def _parse_eval_args(self, data: Dict[Component, Any]) -> Dict[str, Any]: + get = lambda name: data[self.manager.get_elem_by_name(name)] + user_config = load_config() + + if get("top.checkpoints"): + checkpoint_dir = ",".join([ + get_save_dir(get("top.model_name"), get("top.finetuning_type"), ckpt) for ckpt in get("top.checkpoints") + ]) + output_dir = get_save_dir( + get("top.model_name"), get("top.finetuning_type"), "eval_" + "_".join(get("top.checkpoints")) + ) + else: + checkpoint_dir = None + output_dir = get_save_dir(get("top.model_name"), get("top.finetuning_type"), "eval_base") + + args = dict( + stage="sft", + model_name_or_path=get("top.model_path"), + do_eval=True, + predict_with_generate=True, + cache_dir=user_config.get("cache_dir", None), + checkpoint_dir=checkpoint_dir, + finetuning_type=get("top.finetuning_type"), + quantization_bit=int(get("top.quantization_bit")) if get("top.quantization_bit") in ["8", "4"] else None, + template=get("top.template"), + system_prompt=get("top.system_prompt"), + flash_attn=get("top.flash_attn"), + shift_attn=get("top.shift_attn"), + rope_scaling=get("top.rope_scaling") if get("top.rope_scaling") in ["linear", "dynamic"] else None, + dataset_dir=get("eval.dataset_dir"), + dataset=",".join(get("eval.dataset")), + cutoff_len=get("eval.cutoff_len"), + max_samples=int(get("eval.max_samples")), + per_device_eval_batch_size=get("eval.batch_size"), + max_new_tokens=get("eval.max_new_tokens"), + top_p=get("eval.top_p"), + temperature=get("eval.temperature"), + output_dir=output_dir + ) + + if get("eval.predict"): + args.pop("do_eval", None) + args["do_predict"] = True + + return args + + def _preview(self, data: Dict[Component, Any], do_train: bool) -> Generator[Tuple[str, Dict[str, Any]], None, None]: + error = self._initialize(data, do_train) + if error: + gr.Warning(error) + yield error, gr.update(visible=False) + else: + args = self._parse_train_args(data) if do_train else self._parse_eval_args(data) + yield gen_cmd(args), gr.update(visible=False) + + def _launch(self, data: Dict[Component, Any], do_train: bool) -> Generator[Tuple[str, Dict[str, Any]], None, None]: + error = self._initialize(data, do_train) + if error: + gr.Warning(error) + yield error, gr.update(visible=False) + else: + args = self._parse_train_args(data) if do_train else self._parse_eval_args(data) + run_kwargs = dict(args=args, callbacks=[self.trainer_callback]) + self.running = True + self.do_train, self.running_data = do_train, data + self.monitor_inputs = dict(lang=data[self.manager.get_elem_by_name("top.lang")], output_dir=args["output_dir"]) + self.thread = Thread(target=run_exp, kwargs=run_kwargs) + self.thread.start() + yield from self.monitor() + + def preview_train(self, data: Dict[Component, Any]) -> Generator[Tuple[str, Dict[str, Any]], None, None]: + yield from self._preview(data, do_train=True) + + def preview_eval(self, data: Dict[Component, Any]) -> Generator[Tuple[str, Dict[str, Any]], None, None]: + yield from self._preview(data, do_train=False) + + def run_train(self, data: Dict[Component, Any]) -> Generator[Tuple[str, Dict[str, Any]], None, None]: + yield from self._launch(data, do_train=True) + + def run_eval(self, data: Dict[Component, Any]) -> Generator[Tuple[str, Dict[str, Any]], None, None]: + yield from self._launch(data, do_train=False) + + def monitor(self) -> Generator[Tuple[str, Dict[str, Any]], None, None]: + lang, output_dir = self.monitor_inputs["lang"], self.monitor_inputs["output_dir"] + while self.thread.is_alive(): + time.sleep(2) + if self.aborted: + yield ALERTS["info_aborting"][lang], gr.update(visible=False) + else: + yield self.logger_handler.log, update_process_bar(self.trainer_callback) + + if self.do_train: + if os.path.exists(os.path.join(output_dir, TRAINING_ARGS_NAME)): + finish_info = ALERTS["info_finished"][lang] + else: + finish_info = ALERTS["err_failed"][lang] + else: + if os.path.exists(os.path.join(output_dir, "all_results.json")): + finish_info = get_eval_results(os.path.join(output_dir, "all_results.json")) + else: + finish_info = ALERTS["err_failed"][lang] + + yield self._finalize(lang, finish_info), gr.update(visible=False) diff --git a/llm_rl/src/llmtuner/webui/utils.py b/llm_rl/src/llmtuner/webui/utils.py new file mode 100644 index 00000000..933d951d --- /dev/null +++ b/llm_rl/src/llmtuner/webui/utils.py @@ -0,0 +1,85 @@ +import os +import json +import gradio as gr +import matplotlib.figure +import matplotlib.pyplot as plt +from typing import TYPE_CHECKING, Any, Dict +from datetime import datetime + +from llmtuner.extras.ploting import smooth +from llmtuner.webui.common import get_save_dir + +if TYPE_CHECKING: + from llmtuner.extras.callbacks import LogCallback + + +def update_process_bar(callback: "LogCallback") -> Dict[str, Any]: + if not callback.max_steps: + return gr.update(visible=False) + + percentage = round(100 * callback.cur_steps / callback.max_steps, 0) if callback.max_steps != 0 else 100.0 + label = "Running {:d}/{:d}: {} < {}".format( + callback.cur_steps, + callback.max_steps, + callback.elapsed_time, + callback.remaining_time + ) + return gr.update(label=label, value=percentage, visible=True) + + +def get_time() -> str: + return datetime.now().strftime('%Y-%m-%d-%H-%M-%S') + + +def can_quantize(finetuning_type: str) -> Dict[str, Any]: + if finetuning_type != "lora": + return gr.update(value="None", interactive=False) + else: + return gr.update(interactive=True) + + +def gen_cmd(args: Dict[str, Any]) -> str: + args.pop("disable_tqdm", None) + args["plot_loss"] = args.get("do_train", None) + cmd_lines = ["CUDA_VISIBLE_DEVICES=0 python src/train_bash.py "] + for k, v in args.items(): + if v is not None and v != "": + cmd_lines.append(" --{} {} ".format(k, str(v))) + cmd_text = "\\\n".join(cmd_lines) + cmd_text = "```bash\n{}\n```".format(cmd_text) + return cmd_text + + +def get_eval_results(path: os.PathLike) -> str: + with open(path, "r", encoding="utf-8") as f: + result = json.dumps(json.load(f), indent=4) + return "```json\n{}\n```\n".format(result) + + +def gen_plot(base_model: str, finetuning_type: str, output_dir: str) -> matplotlib.figure.Figure: + if not base_model: + return + log_file = get_save_dir(base_model, finetuning_type, output_dir, "trainer_log.jsonl") + if not os.path.isfile(log_file): + return + + plt.close("all") + fig = plt.figure() + ax = fig.add_subplot(111) + steps, losses = [], [] + with open(log_file, "r", encoding="utf-8") as f: + for line in f: + log_info = json.loads(line) + if log_info.get("loss", None): + steps.append(log_info["current_steps"]) + losses.append(log_info["loss"]) + + if len(losses) == 0: + return None + + ax.plot(steps, losses, alpha=0.4, label="original") + ax.plot(steps, smooth(losses), label="smoothed") + ax.legend() + ax.set_xlabel("step") + ax.set_ylabel("loss") + return fig diff --git a/llm_rl/src/train_bash.py b/llm_rl/src/train_bash.py new file mode 100644 index 00000000..9ddd0586 --- /dev/null +++ b/llm_rl/src/train_bash.py @@ -0,0 +1,14 @@ +from llmtuner import run_exp + + +def main(): + run_exp() + + +def _mp_fn(index): + # For xla_spawn (TPUs) + main() + + +if __name__ == "__main__": + main() diff --git a/llm_rl/src/train_web.py b/llm_rl/src/train_web.py new file mode 100644 index 00000000..38efd64d --- /dev/null +++ b/llm_rl/src/train_web.py @@ -0,0 +1,11 @@ +from llmtuner import create_ui + + +def main(): + demo = create_ui() + demo.queue() + demo.launch(server_name="0.0.0.0", server_port=7860, share=False, inbrowser=True) + + +if __name__ == "__main__": + main() diff --git a/llm_rl/src/web_demo.py b/llm_rl/src/web_demo.py new file mode 100644 index 00000000..257536ab --- /dev/null +++ b/llm_rl/src/web_demo.py @@ -0,0 +1,11 @@ +from llmtuner import create_web_demo + + +def main(): + demo = create_web_demo() + demo.queue() + demo.launch(server_name="0.0.0.0", server_port=7860, share=False, inbrowser=True) + + +if __name__ == "__main__": + main() diff --git a/llm_rl/tests/cal_flops.py b/llm_rl/tests/cal_flops.py new file mode 100644 index 00000000..01b005af --- /dev/null +++ b/llm_rl/tests/cal_flops.py @@ -0,0 +1,44 @@ +# coding=utf-8 +# Calculates the flops of pre-trained models. +# Usage: python cal_flops.py --model_name_or_path path_to_model --batch_size 1 --seq_length 512 +# Inspired by: https://www.deepspeed.ai/tutorials/flops-profiler/ + +import fire +import torch +from typing import Optional +from deepspeed.accelerator import get_accelerator # type: ignore +from deepspeed.profiling.flops_profiler import get_model_profile # type: ignore + +from llmtuner import ChatModel + + +def calculate( + model_name_or_path: str, + batch_size: Optional[int] = 1, + seq_length: Optional[int] = 256, + flash_attn: Optional[bool] = False +): + with get_accelerator().device(0): + chat_model = ChatModel(dict( + model_name_or_path=model_name_or_path, + template="vanilla", + flash_attn=flash_attn + )) + fake_input = torch.ones((batch_size, seq_length), dtype=torch.long, device=chat_model.model.device) + input_dict = { + "input_ids": fake_input, + "labels": fake_input.clone() + } + flops, macs, params = get_model_profile( + chat_model.model, + kwargs=input_dict, + print_profile=True, + detailed=True + ) + print("FLOPs:", flops) + print("MACs:", macs) + print("Params:", params) + + +if __name__ == "__main__": + fire.Fire(calculate) diff --git a/llm_rl/tests/llamafy_baichuan2.py b/llm_rl/tests/llamafy_baichuan2.py new file mode 100644 index 00000000..d08eee1c --- /dev/null +++ b/llm_rl/tests/llamafy_baichuan2.py @@ -0,0 +1,86 @@ +# coding=utf-8 +# Converts the Baichuan2-7B model in the same format as LLaMA2-7B. +# Usage: python llamafy_baichuan2.py --input_dir input --output_dir output --shard_size 10GB +# Inspired by: https://huggingface.co/fireballoon/baichuan-llama-7b/blob/main/convert_baichuan_to_llama.py +# Converted model: https://huggingface.co/hiyouga/Baichuan2-7B-Base-LLaMAfied + +import os +import fire +import json +import torch +from collections import OrderedDict +from transformers.modeling_utils import shard_checkpoint, WEIGHTS_NAME, WEIGHTS_INDEX_NAME +from typing import Any, Dict + + +CONFIG_NAME = "config.json" + + +def save_weight( + input_dir: str, + output_dir: str, + shard_size: str +): + baichuan2_state_dict: Dict[str, torch.Tensor] = OrderedDict() + for filepath in os.listdir(input_dir): + if os.path.isfile(os.path.join(input_dir, filepath)) and filepath.endswith(".bin"): + shard_weight = torch.load(os.path.join(input_dir, filepath), map_location="cpu") + baichuan2_state_dict.update(shard_weight) + + llama2_state_dict: Dict[str, torch.Tensor] = OrderedDict() + for key, value in baichuan2_state_dict.items(): + if "W_pack" in key: + proj_size = value.size(0) // 3 + llama2_state_dict[key.replace("W_pack", "q_proj")] = value[:proj_size, :] + llama2_state_dict[key.replace("W_pack", "k_proj")] = value[proj_size:2*proj_size, :] + llama2_state_dict[key.replace("W_pack", "v_proj")] = value[2*proj_size:, :] + elif "lm_head" in key: + llama2_state_dict[key] = torch.nn.functional.normalize(value) + else: + llama2_state_dict[key] = value + + shards, index = shard_checkpoint(llama2_state_dict, max_shard_size=shard_size, weights_name=WEIGHTS_NAME) + for shard_file, shard in shards.items(): + torch.save(shard, os.path.join(output_dir, shard_file)) + + if index is None: + print("Model weights saved in {}".format(os.path.join(output_dir, WEIGHTS_NAME))) + else: + with open(os.path.join(output_dir, WEIGHTS_INDEX_NAME), "w", encoding="utf-8") as f: + json.dump(index, f, indent=2, sort_keys=True) + print("Model weights saved in {}".format(output_dir)) + + +def save_config( + input_dir: str, + output_dir: str +): + with open(os.path.join(input_dir, CONFIG_NAME), "r", encoding="utf-8") as f: + llama2_config_dict: Dict[str, Any] = json.load(f) + + llama2_config_dict["architectures"] = ["LlamaForCausalLM"] + llama2_config_dict.pop("auto_map", None) + llama2_config_dict.pop("tokenizer_class", None) + llama2_config_dict["model_type"] = "llama" + + with open(os.path.join(output_dir, CONFIG_NAME), "w", encoding="utf-8") as f: + json.dump(llama2_config_dict, f, indent=2) + print("Model config saved in {}".format(os.path.join(output_dir, CONFIG_NAME))) + + +def llamafy_baichuan2( + input_dir: str, + output_dir: str, + shard_size: str +): + try: + os.makedirs(output_dir, exist_ok=False) + except Exception as e: + raise print("Output dir already exists", e) + + save_weight(input_dir, output_dir, shard_size) + save_config(input_dir, output_dir) + + +if __name__ == "__main__": + fire.Fire(llamafy_baichuan2) diff --git a/llm_rl/tests/llamafy_qwen.py b/llm_rl/tests/llamafy_qwen.py new file mode 100644 index 00000000..8b9fc395 --- /dev/null +++ b/llm_rl/tests/llamafy_qwen.py @@ -0,0 +1,135 @@ +# coding=utf-8 +# Converts the Qwen models in the same format as LLaMA2. +# Usage: python llamafy_qwen.py --input_dir input --output_dir output --shard_size 10GB + +import os +import fire +import json +import torch +from collections import OrderedDict +from safetensors import safe_open +from transformers.modeling_utils import shard_checkpoint, WEIGHTS_NAME, WEIGHTS_INDEX_NAME +from transformers.utils import check_min_version +from typing import Any, Dict + +try: + check_min_version("4.34.0") +except: + raise ValueError("Please upgrade `transformers` to 4.34.0") + + +CONFIG_NAME = "config.json" + + +def save_weight( + input_dir: str, + output_dir: str, + shard_size: str +) -> str: + qwen_state_dict: Dict[str, torch.Tensor] = OrderedDict() + for filepath in os.listdir(input_dir): + if os.path.isfile(os.path.join(input_dir, filepath)) and filepath.endswith(".safetensors"): + with safe_open(os.path.join(input_dir, filepath), framework="pt", device="cpu") as f: + for key in f.keys(): + qwen_state_dict[key] = f.get_tensor(key) + + llama2_state_dict: Dict[str, torch.Tensor] = OrderedDict() + torch_dtype = None + for key, value in qwen_state_dict.items(): + if torch_dtype is None: + torch_dtype = value.dtype + if "wte" in key: + llama2_state_dict["model.embed_tokens.weight"] = value + elif "ln_f" in key: + llama2_state_dict["model.norm.weight"] = value + else: + key = key.replace("transformer.h", "model.layers") + if "attn.c_attn" in key: + proj_size = value.size(0) // 3 + llama2_state_dict[key.replace("attn.c_attn", "self_attn.q_proj")] = value[:proj_size, ...] + llama2_state_dict[key.replace("attn.c_attn", "self_attn.k_proj")] = value[proj_size:2*proj_size, ...] + llama2_state_dict[key.replace("attn.c_attn", "self_attn.v_proj")] = value[2*proj_size:, ...] + elif "attn.c_proj" in key: + llama2_state_dict[key.replace("attn.c_proj", "self_attn.o_proj")] = value + llama2_state_dict[key.replace("attn.c_proj.weight", "self_attn.o_proj.bias")] = ( + torch.zeros_like(value[:, 0]).squeeze() + ) + elif "ln_1" in key: + llama2_state_dict[key.replace("ln_1", "input_layernorm")] = value + elif "ln_2" in key: + llama2_state_dict[key.replace("ln_2", "post_attention_layernorm")] = value + elif "mlp.w1" in key: + llama2_state_dict[key.replace("mlp.w1", "mlp.up_proj")] = value + elif "mlp.w2" in key: + llama2_state_dict[key.replace("mlp.w2", "mlp.gate_proj")] = value + elif "mlp.c_proj" in key: + llama2_state_dict[key.replace("mlp.c_proj", "mlp.down_proj")] = value + elif "lm_head" in key: + llama2_state_dict[key] = value + else: + raise KeyError("Unable to process key {}".format(key)) + + shards, index = shard_checkpoint(llama2_state_dict, max_shard_size=shard_size, weights_name=WEIGHTS_NAME) + for shard_file, shard in shards.items(): + torch.save(shard, os.path.join(output_dir, shard_file)) + + if index is None: + print("Model weights saved in {}".format(os.path.join(output_dir, WEIGHTS_NAME))) + else: + with open(os.path.join(output_dir, WEIGHTS_INDEX_NAME), "w", encoding="utf-8") as f: + json.dump(index, f, indent=2, sort_keys=True) + print("Model weights saved in {}".format(output_dir)) + + return str(torch_dtype).replace("torch.", "") + + +def save_config( + input_dir: str, + output_dir: str, + torch_dtype: str +): + with open(os.path.join(input_dir, CONFIG_NAME), "r", encoding="utf-8") as f: + qwen_config_dict: Dict[str, Any] = json.load(f) + + llama2_config_dict: Dict[str, Any] = OrderedDict() + llama2_config_dict["architectures"] = ["LlamaForCausalLM"] + llama2_config_dict["hidden_act"] = "silu" + llama2_config_dict["hidden_size"] = qwen_config_dict["hidden_size"] + llama2_config_dict["initializer_range"] = qwen_config_dict["initializer_range"] + llama2_config_dict["intermediate_size"] = qwen_config_dict["intermediate_size"] // 2 + llama2_config_dict["max_position_embeddings"] = qwen_config_dict["max_position_embeddings"] + llama2_config_dict["model_type"] = "llama" + llama2_config_dict["num_attention_heads"] = qwen_config_dict["num_attention_heads"] + llama2_config_dict["num_hidden_layers"] = qwen_config_dict["num_hidden_layers"] + llama2_config_dict["num_key_value_heads"] = qwen_config_dict["hidden_size"] // qwen_config_dict["kv_channels"] + llama2_config_dict["pretraining_tp"] = 1 + llama2_config_dict["rms_norm_eps"] = qwen_config_dict["layer_norm_epsilon"] + llama2_config_dict["rope_scaling"] = None + llama2_config_dict["tie_word_embeddings"] = qwen_config_dict["tie_word_embeddings"] + llama2_config_dict["torch_dtype"] = torch_dtype + llama2_config_dict["transformers_version"] = "4.34.0" + llama2_config_dict["use_cache"] = True + llama2_config_dict["vocab_size"] = qwen_config_dict["vocab_size"] + llama2_config_dict["attention_bias"] = True + + with open(os.path.join(output_dir, CONFIG_NAME), "w", encoding="utf-8") as f: + json.dump(llama2_config_dict, f, indent=2) + print("Model config saved in {}".format(os.path.join(output_dir, CONFIG_NAME))) + + +def llamafy_qwen( + input_dir: str, + output_dir: str, + shard_size: str +): + try: + os.makedirs(output_dir, exist_ok=False) + except Exception as e: + raise print("Output dir already exists", e) + + torch_dtype = save_weight(input_dir, output_dir, shard_size) + save_config(input_dir, output_dir, torch_dtype) + + +if __name__ == "__main__": + fire.Fire(llamafy_qwen) diff --git a/llm_rl/tests/quantize.py b/llm_rl/tests/quantize.py new file mode 100644 index 00000000..25321cf3 --- /dev/null +++ b/llm_rl/tests/quantize.py @@ -0,0 +1,50 @@ +# coding=utf-8 +# Quantizes models with AutoGPTQ (https://github.com/PanQiWei/AutoGPTQ). +# Usage: python quantize.py --input_dir path_to_llama_model --output_dir path_to_quant_model --data_file alpaca.json +# --max_length 1024 --max_samples 1024 +# dataset format: instruction (string), input (string), output (string), history (List[string]) + + +import fire +from datasets import load_dataset +from transformers import AutoTokenizer +from auto_gptq import AutoGPTQForCausalLM, BaseQuantizeConfig + + +def quantize(input_dir: str, output_dir: str, data_file: str, max_length: int, max_samples: int): + tokenizer = AutoTokenizer.from_pretrained(input_dir, use_fast=False, padding_side="left") + + def format_example(examples): + prefix=("A chat between a curious user and an artificial intelligence assistant. " + "The assistant gives helpful, detailed, and polite answers to the user's questions.") + texts = [] + for i in range(len(examples["instruction"])): + prompt = prefix + "\n" + if "history" in examples: + for user_query, bot_resp in examples["history"][i]: + prompt += "Human: {}\nAssistant: {}\n".format(user_query, bot_resp) + prompt += "Human: {}\nAssistant: {}".format( + examples["instruction"][i] + "\n" + examples["input"][i], examples["output"][i] + ) + texts.append(prompt) + return tokenizer(texts, truncation=True, max_length=max_length) + + dataset = load_dataset("json", data_files=data_file)["train"] + column_names = list(dataset.column_names) + dataset = dataset.select(range(min(len(dataset), max_samples))) + dataset = dataset.map(format_example, batched=True, remove_columns=column_names) + dataset = dataset.shuffle() + + quantize_config = BaseQuantizeConfig( + bits=4, + group_size=128, + desc_act=False + ) + + model = AutoGPTQForCausalLM.from_pretrained(input_dir, quantize_config, trust_remote_code=True) + model.quantize(dataset) + model.save_quantized(output_dir) + + +if __name__ == "__main__": + fire.Fire(quantize)