Quick Start
Let's begin Prompting in 1 minute
Installation with pip install
Directly install from PyPI:
pip install promplate[openai]
We are using OpenAI
just for demonstration. In fact, you can use any LLM as you want.
Make LLM Calls
First, Open a python REPL 💻 (ipython
or jupyter
are OK. Just any REPL you like)
All the code below should run “as is”, which means you can copy and paste them in your terminal and it will work fine.
>>> from promplate.llm.openai import ChatComplete # this simply wraps OpenAI's SDK >>> complete = ChatComplete(api_key="...")
The
api_key
should be filled with your API Key from OpenAI Platform
Then call it like this:
>>> complete("hi", model="gpt-3.5-turbo-1106") 'Hello! How can I assist you today?'
Perhaps you don’t want to provide model
parameter everytime you call complete
, so you can bind it like this:
>>> complete = ChatComplete(api_key="...").bind(model="gpt-3.5-turbo-1106")
Then call it simply with a string:
>>> complete("hi", model="gpt-3.5-turbo-1106") 'Hello! How can I assist you today?'
If you don't have an OpenAI API Key 🔑
You could use our FREE proxy site asbase_url
like this:
>>> from promplate.llm.openai import ChatComplete >>> from httpx import Client # or AsyncClient if use use AsyncChatComplete >>> client = Client(follow_redirects=True) # this must be set because our free api uses 308 redirects >>> complete = ChatComplete(base_url="https://promplate.dev", api_key="<any-value>", http_client=client).bind(model="gpt-3.5-turbo-1106") >>> complete("hi") 'Hello! How can I assist you today?'
If you want to use instruct models 🤔
Simply replaceChatComplete
by TextComplete
:
>>> from promplate.llm.openai import TextComplete >>> complete = TextComplete(api_key="...").bind(model="gpt-3.5-turbo-instruct") >>> complete("I am") ' just incredibly proud of the team, and their creation of a brand new ship makes'
Complete
instance:
>>> complete("1 + 1 = ", temperature=0, max_tokens=1) '2'
If you prefer to stream the response 👀
It is still super easy, just useChatGenerate
:
>>> from promplate.llm.openai import ChatGenerate >>> generate = ChatGenerate(api_key="...").bind(model="gpt-3.5-turbo-1106") >>> for i in generate("Explain why 1 + 1 = 2"): ... print(i, end="", flush=True) # this will print generated tokens gradually ... The equation 1 + 1 = 2 is a fundamental principle in mathematics and arithmetic. It represents the addition operation, which involves combining two quantities or numbers to find their sum. In this case, when we add 1 to another 1, we are essentially combining or merging two individual units or quantities. By doing this, we end up with a total count of two. Therefore, the result is 2. This principle is consistent and holds true in all contexts and across different number systems, whether it is in the base-10 decimal system, binary system, or any other number system. 1 + 1 = 2 is considered a basic and universally accepted mathematical fact, forming the foundation for more complex mathematical operations and calculations.
Prompting with Template
There must be something dynamic in your prompt, like user queries, retrieved context, search results, etc.
In promplate, simply use {{ }}
to insert dynamic data.
>>> import time >>> from promplate import Template >>> greet = Template("Greet me. It is {{ time.asctime() }} now.") >>> greet.render(locals()) 'Greet me. It is Sun Oct 1 03:56:02 2023 now.'
You can run the prompt by complete
we created before
>>> complete(_) 'Good morning!'
Wow, it works fine. In fact, you can use any python expression inside {{ }}
.
Tips: you can combine partial context to a template like this:
>>> import time >>> from promplate import Template >>> greet = Template("Greet me. It is {{ time.asctime() }} now.", {"time": time}) # of course you can use locals() here too >>> greet.render() # empty parameter is ok 'Greet me. It is Sun Oct 1 03:56:02 2023 now.'
Turning a complex task into small pieces
Sometimes we don’t use a single prompt to complete. Here are some reasons:
- Describing a complex task in a single prompt may be difficult
- Splitting big task into small ones maybe can reduce the total token usage
- If you need structural output, it is easier to specifying data formats separately
- We human can think quicker after breaking task into parts
- Breaking big tasks into sub tasks may enhance interpretability, reducing debugging time
- …
In prompate
, We use a Node
to represent a single “task”. You can initialize a Task
with a string like initiating a Template
:
>>> from promplate import Node >>> greet = Node("Greet me. It is {{ time.asctime() }} now.", locals()) >>> greet.render() 'Greet me. It is Sun Oct 1 04:16:04 2023 now.'
But there are far more utilities.
Such as, you can add two nodes together magically:
>>> translate = Node('translate """{{ __result__ }}""" into {{ target_language }}') >>> chain = greet + translate # this represents the pipeline of "greeting in another language" >>> chain.invoke({"target_language": "zh_CN"}, complete).result '早上好!'
Details
Mention that the return type of .invoke()
is ChainContext
which combines the context passed everywhere in a right order.
__result__
is the output of last Node
. It is automatically assigned during .invoke()
. You can access it inside the template.
Outside the template, you can use .result
of a ChainContext
to get the last output.
The following three expressions should return the same string:
>>> template = Template("...") >>> complete(template.render())
>>> Node("...").invoke()["__result__"]
>>> Node("...").invoke().result
This part may be a bit more complex, but believe me, this enhances the flexibility of this framework and you will like it.
Registering callbacks
There are sometimes some work can’t be done without our code. such as:
- LLM returns string, while we may want to parse it into structural data format like
dict
orlist
- We may need to log the intermediate variables to see whether the previous nodes work fine
- We may modify context dynamically during a chain running
- …
In promplate
, you can register a callback everytime before or after a node runs.
Besides manually implementing the Callback
interface, you can directly use decorators syntax like so:
>>> @greet.end_process ... def log_greet_result(context): ... print(context.result) ... print(context["target_language"]) ... >>> chain.invoke({"target_language": "zh_CN"}, complete).result Good morning! zh_CN '早上好!'
Congratulations 🎉 You’ve learnt the basic paradigm of using promplate
for prompt engineering.
Thanks for reading. There are still lots of features not mentioned here. Learn more in other pages 🤗 If you have any questions, please feel free to ask us on GitHub Discussions.