Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: continuous batching with vLLM #349

Merged
merged 5 commits into from
Sep 14, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
feat: continuous batching
Signed-off-by: paperspace <29749331+aarnphm@users.noreply.github.com>
  • Loading branch information
aarnphm committed Sep 14, 2023
commit c9cf103474b5b62e1d41e2e04730a2ce3bbdaf19
139 changes: 139 additions & 0 deletions bench.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,139 @@
from __future__ import annotations
import asyncio
import json

import aiohttp

import openllm

async def send_request(url, prompt, session, model, **attrs):
headers = {'accept': 'application/json', 'Content-Type': 'application/json'}
config = openllm.AutoConfig.for_model(model).model_construct_env(**attrs).model_dump()
data = {'prompt': prompt, 'llm_config': config, 'adapter_name': None}
async with session.post(url, headers=headers, data=json.dumps(data)) as response:
result = await response.text()
print('-' * 10 + '\n\n prompt:', prompt, '\nGeneration:', result, '\n\n' + '-' * 10)

async def main():
url = 'http://localhost:3000/v1/generate_stream'
# len=100
prompts = [
'What is the meaning of life?',
'Explain the concept of quantum entanglement.',
'Describe the process of photosynthesis.',
'What are the benefits of regular exercise?',
'How does the internet work?',
'Discuss the impact of climate change on ecosystems.',
'Explain the principles of supply and demand in economics.',
'What is the history of the Roman Empire?',
'Describe the structure of a cell.',
'Discuss the pros and cons of renewable energy sources.',
'Explain the theory of relativity.',
'What is the role of DNA in genetics?',
'Describe the art movement of the Renaissance.',
'Discuss the causes of World War I.',
'What are the major functions of the human brain?',
'Explain the process of evolution by natural selection.',
'Describe the cultural significance of the Great Wall of China.',
'What is the impact of social media on society?',
'Discuss the life and works of Shakespeare.',
'Explain the concept of artificial intelligence.',
'What are the different types of chemical reactions?',
"Describe the structure of the Earth's atmosphere.",
'Discuss the history of the civil rights movement.',
'What are the economic implications of globalization?',
'Explain the principles of good nutrition.',
'Describe the major functions of the immune system.',
'Discuss the impact of colonialism on Africa.',
'What is the process of cellular respiration?',
'Explain the importance of biodiversity.',
'Discuss the causes and consequences of the Industrial Revolution.',
'What are the fundamental principles of democracy?',
'Describe the major components of a computer.',
'Explain the concept of human rights.',
'What is the role of enzymes in biological reactions?',
'Discuss the history of space exploration.',
'What are the ethical considerations in medical research?',
'Describe the cultural significance of the Pyramids of Egypt.',
'Explain the principles of classical physics.',
'What is the impact of climate change on weather patterns?',
'Discuss the major events of the American Revolution.',
'What are the effects of pollution on the environment?',
'Describe the process of protein synthesis.',
'Explain the concept of sustainable agriculture.',
'What is the history of the European Union?',
'Discuss the impact of the Renaissance on art and culture.',
'What are the key principles of marketing?',
'Explain the structure of the periodic table.',
'Describe the major types of renewable energy.',
'Discuss the causes and consequences of the French Revolution.',
'What is the role of the United Nations in international relations?',
'Explain the principles of game theory in economics.',
'What are the stages of human development?',
'Describe the cultural significance of the Taj Mahal.',
'Discuss the major themes in the works of Ernest Hemingway.',
'What is the impact of automation on the workforce?',
'Explain the concept of genetic engineering.',
'What are the different types of chemical bonds?',
"Describe the layers of the Earth's atmosphere.",
"Discuss the history of the women's suffrage movement.",
'What are the economic factors influencing consumer behavior?',
'Explain the principles of conflict resolution.',
'What is the role of neurotransmitters in the nervous system?',
'Discuss the impact of colonialism on India.',
'What is the process of mitosis?',
'Explain the importance of water conservation.',
'Describe the cultural significance of the Acropolis in Athens.',
'Discuss the major philosophical ideas of Plato.',
'What are the principles of investment in finance?',
'Explain the structure of a virus.',
'What is the history of the United Nations?',
'Discuss the impact of technology on modern art.',
'What are the key concepts in cognitive psychology?',
'Describe the major types of non-renewable energy sources.',
'Explain the causes and consequences of the Russian Revolution.',
'What is the role of the World Health Organization in global health?',
'Discuss the principles of ethics in business.',
'What are the stages of the water cycle?',
'Explain the concept of social justice.',
'What is the impact of deforestation on climate change?',
'Describe the process of meiosis.',
'Discuss the cultural significance of the Sistine Chapel ceiling.',
'What are the major themes in the novels of Jane Austen?',
'Explain the role of branding in marketing.',
'What is the history of the Internet?',
'Discuss the impact of artificial intelligence on society.',
'What are the principles of statistical analysis in research?',
'Explain the structure of an atom.',
'What is the significance of the Theory of Evolution by Charles Darwin?',
'Describe the major types of renewable energy.',
'Discuss the causes and consequences of the American Civil War.',
'What is the role of the International Monetary Fund in global economics?',
'Explain the principles of environmental conservation.',
'What are the stages of the rock cycle?',
'Describe the concept of cultural relativism.',
'Discuss the major contributions of Leonardo da Vinci to art and science.',
'What is the impact of globalization on cultural diversity?',
'Explain the process of genetic inheritance.',
'What are the different forms of government in the world?',
'Describe the major types of pollution.',
'Discuss the history of the labor movement.',
'What are the principles of sustainable urban planning?',
'Explain the role of hormones in the endocrine system.',
'What is the cultural significance of the Great Barrier Reef?',
'Discuss the major ideas of Friedrich Nietzsche.',
'What is the impact of social media on political movements?',
'Explain the concept of quantum computing.',
'What are the principles of international diplomacy?',
'Describe the major types of ocean ecosystems.',
'Discuss the causes and consequences of the Cold War.',
'What is the role of the World Trade Organization in global trade?',
'Explain the principles of behavioral psychology.',
'What are the stages of the nitrogen cycle?',
'Describe the concept of cultural appropriation.',
'Discuss the major works of Vincent van Gogh.',
]
async with aiohttp.ClientSession() as session:
await asyncio.gather(*[send_request(url, prompt, session, 'llama', max_new_tokens=4096, top_p=0.21) for _, prompt in enumerate(prompts)])

if __name__ == '__main__': asyncio.run(main())
8 changes: 0 additions & 8 deletions openllm-core/src/openllm_core/_configuration.py
Original file line number Diff line number Diff line change
Expand Up @@ -465,7 +465,6 @@ class ModelSettings(t.TypedDict, total=False):

# meta
url: str
requires_gpu: bool
trust_remote_code: bool
service_name: NotRequired[str]
requirements: t.Optional[ListStr]
Expand Down Expand Up @@ -523,7 +522,6 @@ def default(cls) -> _ModelSettingsAttr:
'cpu': 'pt', 'nvidia.com/gpu': 'pt'
},
name_type='dasherize',
requires_gpu=False,
url='',
model_type='causal_lm',
trust_remote_code=False,
Expand All @@ -541,7 +539,6 @@ def default(cls) -> _ModelSettingsAttr:
architecture: str
default_backend: t.Dict[LiteralResourceSpec, LiteralBackend]
url: str
requires_gpu: bool
trust_remote_code: bool
service_name: str
requirements: t.Optional[ListStr]
Expand Down Expand Up @@ -737,8 +734,6 @@ def __attrs_init__(self, *args: t.Any, **attrs: t.Any) -> None:
'''The default backend to run LLM based on available accelerator. By default, it will be PyTorch (pt) for most models. For some models, such as Llama, it will use `vllm` or `flax`. It is a dictionary of key as the accelerator spec in k8s ('cpu', 'nvidia.com/gpu', 'amd.com/gpu', 'cloud-tpus.google.com/v2', ...) and the values as supported OpenLLM backend ('flax', 'tf', 'pt', 'vllm', 'ggml', 'mlc')'''
__openllm_url__: str = Field(None)
'''The resolved url for this LLMConfig.'''
__openllm_requires_gpu__: bool = Field(None)
'''Determines if this model is only available on GPU. By default it supports GPU and fallback to CPU.'''
__openllm_trust_remote_code__: bool = Field(None)
'''Whether to always trust remote code'''
__openllm_service_name__: str = Field(None)
Expand Down Expand Up @@ -932,7 +927,6 @@ class FalconConfig(openllm.LLMConfig):
__config__ = {
"name_type": "lowercase",
"trust_remote_code": True,
"requires_gpu": True,
"timeout": 3600000,
"url": "https://falconllm.tii.ae/",
"requirements": ["einops", "xformers", "safetensors"],
Expand Down Expand Up @@ -1108,8 +1102,6 @@ def __getitem__(self, item: t.Literal['default_backend']) -> t.Dict[LiteralResou
@overload
def __getitem__(self, item: t.Literal['url']) -> str: ...
@overload
def __getitem__(self, item: t.Literal['requires_gpu']) -> bool: ...
@overload
def __getitem__(self, item: t.Literal['trust_remote_code']) -> bool: ...
@overload
def __getitem__(self, item: t.Literal['service_name']) -> str: ...
Expand Down
4 changes: 3 additions & 1 deletion openllm-core/src/openllm_core/_typing_compat.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@

from .utils.lazy import VersionInfo

M = t.TypeVar('M', bound='t.Union[transformers.PreTrainedModel, transformers.Pipeline, transformers.TFPreTrainedModel, transformers.FlaxPreTrainedModel, vllm.LLMEngine, peft.PeftModel]')
M = t.TypeVar('M', bound='t.Union[transformers.PreTrainedModel, transformers.Pipeline, transformers.TFPreTrainedModel, transformers.FlaxPreTrainedModel, vllm.AsyncLLMEngine, peft.PeftModel]')
T = t.TypeVar('T', bound='t.Union[transformers.PreTrainedTokenizerFast, transformers.PreTrainedTokenizer, transformers.PreTrainedTokenizerBase]')

def get_literal_args(typ: t.Any) -> tuple[str, ...]:
Expand Down Expand Up @@ -95,6 +95,7 @@ class LLMRunnable(bentoml.Runnable, t.Generic[M, T]):
generate: RunnableMethod[LLMRunnable[M, T], [str], list[t.Any]]
generate_one: RunnableMethod[LLMRunnable[M, T], [str, list[str]], t.Sequence[dict[t.Literal['generated_text'], str]]]
generate_iterator: RunnableMethod[LLMRunnable[M, T], [str], t.Iterator[t.Any]]
vllm_generate_iterator: RunnableMethod[LLMRunnable[M, T], [str], t.AsyncGenerator[str, None]]

class LLMRunner(bentoml.Runner, t.Generic[M, T]):
__doc__: str
Expand All @@ -112,6 +113,7 @@ class LLMRunner(bentoml.Runner, t.Generic[M, T]):
generate: RunnerMethod[LLMRunnable[M, T], [str], list[t.Any]]
generate_one: RunnerMethod[LLMRunnable[M, T], [str, list[str]], t.Sequence[dict[t.Literal['generated_text'], str]]]
generate_iterator: RunnerMethod[LLMRunnable[M, T], [str], t.Iterator[t.Any]]
vllm_generate_iterator: RunnerMethod[LLMRunnable[M, T], [str], t.AsyncGenerator[str, None]]

def __init__(self,
runnable_class: type[LLMRunnable[M, T]],
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,6 @@ class BaichuanConfig(openllm_core.LLMConfig):
'name_type': 'lowercase',
'trust_remote_code': True,
'timeout': 3600000,
'requires_gpu': True,
'url': 'https://github.com/baichuan-inc/Baichuan-7B',
'requirements': ['cpm-kernels', 'sentencepiece'],
'architecture': 'BaiChuanForCausalLM',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,6 @@ class ChatGLMConfig(openllm_core.LLMConfig):
'name_type': 'lowercase',
'trust_remote_code': True,
'timeout': 3600000,
'requires_gpu': True,
'url': 'https://github.com/THUDM/ChatGLM-6B',
'requirements': ['cpm-kernels', 'sentencepiece'],
'architecture': 'ChatGLMForConditionalGeneration',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,6 @@ class FalconConfig(openllm_core.LLMConfig):
__config__ = {
'name_type': 'lowercase',
'trust_remote_code': True,
'requires_gpu': True,
'timeout': int(36e6),
'url': 'https://falconllm.tii.ae/',
'requirements': ['einops', 'xformers'],
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,6 @@ class GPTNeoXConfig(openllm_core.LLMConfig):
__config__ = {
'model_name': 'gpt_neox',
'start_name': 'gpt-neox',
'requires_gpu': True,
'architecture': 'GPTNeoXForCausalLM',
'url': 'https://github.com/EleutherAI/gpt-neox',
'default_id': 'eleutherai/gpt-neox-20b',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
Run a LLMServer for OPT model.
\b
> See more information about falcon at [facebook/opt-66b](https://huggingface.co/facebook/opt-66b)
> See more information about OPT at [facebook/opt-66b](https://huggingface.co/facebook/opt-66b)
\b
## Usage
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,6 @@ class StarCoderConfig(openllm_core.LLMConfig):
"""
__config__ = {
'name_type': 'lowercase',
'requires_gpu': True,
'url': 'https://github.com/bigcode-project/starcoder',
'architecture': 'GPTBigCodeForCausalLM',
'requirements': ['bitsandbytes'],
Expand Down
2 changes: 1 addition & 1 deletion openllm-python/pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -115,7 +115,7 @@ openai = ["openai", "tiktoken"]
opt = ["flax>=0.7", "jax", "jaxlib", "tensorflow", "keras"]
playground = ["jupyter", "notebook", "ipython", "jupytext", "nbformat"]
starcoder = ["bitsandbytes"]
vllm = ["vllm>=0.1.6", "ray"]
vllm = ["vllm>=0.1.7", "ray"]

[tool.hatch.version]
fallback-version = "0.0.0"
Expand Down
10 changes: 5 additions & 5 deletions openllm-python/src/openllm/_assign.py
Original file line number Diff line number Diff line change
Expand Up @@ -44,9 +44,9 @@ def inner(self: LLM[M, T], *decls: t.Any, trust_remote_code: bool | None = None,

return inner

def load_model(fn: load_model_protocol[M, T]) -> t.Callable[[LLM[M, T]], M | vllm.LLMEngine]:
def load_model(fn: load_model_protocol[M, T]) -> t.Callable[[LLM[M, T]], M | vllm.AsyncLLMEngine]:
@functools.wraps(fn)
def inner(self: LLM[M, T], *decls: t.Any, **attrs: t.Any) -> M | vllm.LLMEngine:
def inner(self: LLM[M, T], *decls: t.Any, **attrs: t.Any) -> M | vllm.AsyncLLMEngine:
if self.__llm_backend__ == 'vllm':
num_gpus, dev = 1, device_count()
if dev >= 2: num_gpus = min(dev // 2 * 2, dev)
Expand Down Expand Up @@ -131,10 +131,10 @@ def dunder_support(key: str) -> str:

return codegen.generate_function(cls, '__assign_llm_attr', lines, args=('cls', *args), globs=globs, annotations={'cls': 't.Type[LLM]', 'return': None})

def vllm_postprocess_generate(self: LLM['vllm.LLMEngine', T], prompt: str, generation_result: list[dict[str, t.Any]], **_: t.Any) -> str:
def vllm_postprocess_generate(self: LLM['vllm.AsyncLLMEngine', T], prompt: str, generation_result: list[dict[str, t.Any]], **_: t.Any) -> str:
return generation_result[0]['outputs'][0]['text']

def vllm_generate_iterator(self: LLM['vllm.LLMEngine', T],
def vllm_generate_iterator(self: LLM['vllm.AsyncLLMEngine', T],
prompt: str,
/,
*,
Expand Down Expand Up @@ -164,7 +164,7 @@ def vllm_generate_iterator(self: LLM['vllm.LLMEngine', T],
yield {'text': text_outputs, 'error_code': 0}
if request_output.finished: break

def vllm_generate(self: LLM['vllm.LLMEngine', T], prompt: str, **attrs: t.Any) -> list[dict[str, t.Any]]:
def vllm_generate(self: LLM['vllm.AsyncLLMEngine', T], prompt: str, **attrs: t.Any) -> list[dict[str, t.Any]]:
request_id: str | None = attrs.pop('request_id', None)
if request_id is None: raise ValueError('request_id must not be None.')
outputs: list[vllm.RequestOutput] = []
Expand Down
Loading