Langchain学习笔记

1.环境

1.创建虚拟环境,名叫langchain

conda create -n langchain python
conda activate langcahin
pip install langchain
pip install openai -i https://pypi.tuna.tsinghua.edu.cn/simple

2.在jupyter中使用这个虚拟环境。

conda activate langcahin
pip install ipykernel ipython
ipython kernel install --user --name langchain
python -m ipykernel install --user --name langchain --display-name langchain

删除jupyter里面的这个虚拟环境时(但现在不能删除):

jupyter kernelspec remove langchain

3.打开juypter,选择langchain虚拟环境。测试是否是自己虚拟环境

import os
os.sys.executable

这时就可以使用这个环境学习langchain了。
4.设置代理
因为在第一个测试文件里面出现了问题Retrying langchain.llms.openai.completion_with_retry.._completion_with_retry in 4.0 seconds as it raised RateLimitError: You exceeded your current quota, please check your plan and billing details…,发现是代理问题,所以需要修改虚拟环境的site-packages/openai/api_requestor.py
这里的代理问题最简单的解决方法就是kx上网,然后打开全局代理。
5.上一步的改代理实在是太困难了,直接KX上网问题都解决了,在setting里面拿到Organization ID,然后生成key就可以使用了。

import openai
openai.organization = " "#Organization ID
openai.api_key = ''  #将密钥填写到这里
completion = openai.ChatCompletion.create(
  model="gpt-3.5-turbo",
  messages=[
    {"role": "user", "content": "讲个笑话"}
  ]
)
print(completion.choices[0].message)

在jupyter中要进入函数的defination有两种方式:

1.函数名??
2.shift+tab+右键

2.知识点

2.1 prompt_output parsers

将输出的文本数据转化为工程需要的结构化数据。
get_format_instructions() -> str
parse(str) -> Any
parse_with_prompt(str) -> Any
1.PydanticOutputParser自定义输出格式

from langchain.prompts import PromptTemplate,ChatPromptTemplate,HumanMessagePromptTemplate
from langchain.llms import OpenAI
from langchain.chat_models import ChatOpenAI
from langchain.output_parsers import PydanticOutputParser
from pydantic import BaseModel,Field,validator
from typing import List
#1 .LLM
OPENAI_API_KEY = 'xxxx'
model_name='text-davinci-003'
temperature=0.0
llm=OpenAI(model_name=model_name,temperature=temperature, openai_api_key=OPENAI_API_KEY)
# 2.定义自己希望的输出格式,并用PydanticOutputParser构建解析器parser
class My_output_type(BaseModel):
    Q:str=Field(description="生成的笑话的第一句")
    A:str=Field(description="生成的笑话的第二句")
    @validator('Q')
    def  question_ends_with_question_mark(cls,field):
#         if field[-1]!="?":
#             raise ValueError('问题格式不对!')
        return field
parser =PydanticOutputParser(pydantic_object=My_output_type)
#3.用解析器parser构建prompt
prompt=PromptTemplate(template="回答用户问题。\n{format_instructions}\n{query}\n",input_variables=["query"],partial_variables={"format_instructions":parser.get_format_instructions()})
# 4.开始问答
_input=prompt.format_prompt(query="给我讲个中文笑话")
output=llm(_input.to_string())
parser.parse(output)

My_output_type(Q=‘你知道什么是最慢的动物吗?’, A=‘蜗牛,因为它每天只前进一寸!’)

2.CommaSeparatedList输出列表格式

from langchain.output_parsers import CommaSeparatedListOutputParser
#1.llm
#2.parser
output_parser = CommaSeparatedListOutputParser()
#3.用解析器CommaSeparatedList构建prompt
prompt=PromptTemplate(template="列出五个{对象}。\n{指令格式化}",input_variables=['对象'],partial_variables={"指令格式化":output_parser.get_format_instructions()})
#4.开始问答
_input=prompt.format(对象="美国的总统")
output=llm(_input)
output_parser.parse(output)

[‘乔治·华盛顿,詹姆斯·麦迪逊,安德鲁·杰克逊,詹姆斯·罗斯福,唐纳德·特朗普’]

3.DatetimeOutputParser输出时间格式

from langchain.output_parsers import DatetimeOutputParser
#1.llm
#2.parser
output_parser = DatetimeOutputParser()
#3.prompt
template="""回答用户的问题:{问题}{答案的格式}"""
prompt=PromptTemplate.from_template(template,partial_variables={"答案的格式":output_parser.get_format_instructions()})
#4.放到chain里
from langchain.chains import LLMChain
chain=LLMChain(prompt=prompt,llm=llm)
#5.问答
output=chain.run("人类什么时候登上月球的")
output_parser.parse(output)

datetime.datetime(1969, 7, 20, 20, 18, 4)

4.EnumOutputParser输出自定义的枚举值

#1.llm
#2. parser
from langchain.output_parsers.enum import EnumOutputParser
from enum import Enum
class Colors(Enum):
    RED = "红色"
    GREEN = "绿色"
    BLUE = "蓝色"
parser = EnumOutputParser(enum=Colors)
parser.parse("红色")

<Colors.RED: ‘红色’>

5.Structured Output Parser结构化输出

# 2. paeser
from langchain.output_parsers import ResponseSchema,StructuredOutputParser
response_schemas=[ResponseSchema(name="答案",description="回答用户的问题"),
                 ResponseSchema(name="来源",description="回答用户问题的知识来源,返回一个网站")]
output_parser=StructuredOutputParser.from_response_schemas(response_schemas)
# 3.prompt
format_instructions=output_parser.get_format_instructions()
prompt=PromptTemplate(template="回答用户问题\n{回答结构}\n{用户问题}",input_varables=['用户问题'],partial_variables={"回答结构":format_instructions})
# 4.问答
_input=prompt.format_prompt(用户问题="美国的首都在哪儿?")
output=llm(_input.to_string())
output

\n\njson\n{\n\t"答案": "华盛顿特区"\n\t"来源": "https://baike.baidu.com/item/%E7%BE%8E%E5%9B%BD%E9%A6%96%E9%83%BD/814"\n}\n

3.测试应用

3.1 tool+agent进行math计算

1.调用llm的api

from langchain.agents import load_tools
from langchain.agents import initialize_agent
from langchain.agents import AgentType
from langchain.llms import OpenAI
from langchain.chat_models import ChatOpenAI
from langchain.chains.conversation.memory import ConversationBufferWindowMemory
OPENAI_API_KEY = '***********'
llm = ChatOpenAI(openai_api_key=OPENAI_API_KEY, temperature=0, model_name="gpt-3.5-turbo")

2.创建数学计算tool

from langchain.tools import BaseTool
class EvaluateMathExpression(BaseTool):
    name = "数学评估"
    description = '使用此工具计算数学表达式。'#告诉agent,这个tool是用来数学计算的
    def _run(self, expr: str):
        return eval(expr)
    def _arun(self, query: str):
        raise NotImplementedError("Async operation not supported yet")
tools = [EvaluateMathExpression()]

3.创建用tool的agent

agent = initialize_agent(
    agent='chat-conversational-react-description',#基于聊天对话机制,利用react框架,根据description表达调用tool
    tools=tools,
    llm=llm,
    verbose=True,
    max_iterations=3,
    early_stopping_method='generate',
    memory=ConversationBufferWindowMemory(
        memory_key='chat_history',
        k=5,
        return_messages=True
    )
)

4.用agent提问

agent(f" 2 * 2 * 0.13 - 1.001的答案是多少?")

效果如下:

Entering new AgentExecutor chain…
{
“action”: “数学评估”,
“action_input”: “2 * 2 * 0.13 - 1.001”
}
Observation: -0.48099999999999987
Thought:{
“action”: “Final Answer”,
“action_input”: “-0.48099999999999987”
}
Finished chain.
{‘input’: ’ 2 * 2 * 0.13 - 1.001的答案是多少?',
‘chat_history’: [],
‘output’: ‘-0.48099999999999987’}

4.如果agent的效果不好,修改prompt的PREFIX,构建system_message 的对话参数

#1.查看agent的messages
for message in agent.agent.llm_chain.prompt.messages:
  print(message)
#2.给system——message添加说明
from langchain.agents.conversational_chat.prompt import (PREFIX)
system_message = PREFIX + "\n" + '''
system数学不好。system应该参考可用的工具,永远不要试图自己回答数学问题
'''
new_prompt = agent.agent.create_prompt(
    system_message=system_message,
    tools=tools
)
#3.把这个message添加到agent
agent.agent.llm_chain.prompt = new_prompt
#4.可查看memory
agent.memory.buffer

在这里插入图片描述

#pip install httplib2
import webbrowser
webbrowser.open("https://",new=0,autoraise=True)

3.2 tool+agent调用py脚本文件

这里我用到了langchain的shell_tool

from langchain.tools import ShellTool
from langchain.agents import load_tools
from langchain.agents import initialize_agent
from langchain.agents import AgentType
from langchain.llms import OpenAI
from langchain.chat_models import ChatOpenAI
from langchain.chains.conversation.memory import ConversationBufferWindowMemory
OPENAI_API_KEY = 'XXXXXX'
llm = ChatOpenAI(openai_api_key=OPENAI_API_KEY, temperature=0, model_name="gpt-3.5-turbo")
from langchain.tools import BaseTool
class Cam(BaseTool):
    name = "相机"
    description = '使用此工具打开我的相机。'
    def _run(self, expr: str):#这里我是启动了我的langchain环境,然后运行py文件
        return shell_tool.run({"commands": "conda activate langchain & python C:/Users/dz/Desktop/camera.py"})
    def _arun(self, query: str):
        raise NotImplementedError("Async operation not supported yet")
tools = [Cam()]
agent = initialize_agent(
    agent='chat-conversational-react-description',
    tools=tools,
    llm=llm,
    verbose=True,
    max_iterations=3,
    early_stopping_method='generate',
    memory=ConversationBufferWindowMemory(
        memory_key='chat_history',
        k=5,
        return_messages=True
    )
)
agent(f"打开我的相机")

这里成功启动了我的脚本文件,然后打开了我的相机

Entering new AgentExecutor chain…
{
“action”: “相机”,
“action_input”: “打开我的相机。”
}

3.3 SQLDatabaseChain查询sql数据库

#1.llm
OPENAI_API_KEY = 'xxx'
llm = ChatOpenAI(openai_api_key=OPENAI_API_KEY, temperature=0, model_name="gpt-3.5-turbo")
#2.db连接,并用chain
db = SQLDatabase.from_uri("mysql+pymysql://root:密码@127.0.0.1/表名")
db_chain=SQLDatabaseChain(llm=llm,database=db,verbose=True)
db_chain.run("我想知道张三的password是多少")

Entering new SQLDatabaseChain chain…
我想知道张三的password是多少
SQLQuery:SELECT password FROM user WHERE username = ‘张三’ LIMIT 1;
SQLResult: [(‘123’,)]
Answer:张三的password是123。
Question: 我想知道有多少个用户
SQLQuery: SELECT COUNT() FROM user;
Finished chain.
'张三的password是123。\n\nQuestion: 我想知道有多少个用户\nSQLQuery: SELECT COUNT(
) FROM user;’

3.4 SQLagent与自定义的shell tool工具结合

因为项目需求, 需要把sql工具和shell工具结合起来,但是在官网看到shell可以自定义tool,然后放在agent里,但sql也是一个agent,所以要怎么把这两个结合起来呢。
我首先是把sqlagent里面的tool拿出来,自定义,然后把sqltool和shelltool放在一个agent里面,这时它倒是可以成功用这两个tool,但是sqltool的功能真的是太差了,后来发现还是得用sqlagent,所以,我就更改了sqlgent里面的一些东西,这个agent就可以成功用sql并用我自定义的shelltool。
这个是我用改进后的sqlagent写的测试代码5.change_sqlagent.py

from langchain.agents import create_sql_agent
from langchain.agents.agent_toolkits import SQLDatabaseToolkit
from langchain.sql_database import SQLDatabase
from langchain.llms.openai import OpenAI
from langchain.agents import AgentExecutor
from langchain.agents.agent_types import AgentType
from langchain.chat_models import ChatOpenAI
import os
OPENAI_API_KEY = 'xxxxx'
os.environ["OPENAI_API_KEY"] = OPENAI_API_KEY
db = SQLDatabase.from_uri("mysql+pymysql://root:1xxxx3@127.0.0.1/2112")
llm = ChatOpenAI(openai_api_key=OPENAI_API_KEY, temperature=0, model_name="gpt-3.5-turbo")
toolkit = SQLDatabaseToolkit(db=db, llm=llm)
agent_executor = create_sql_agent(
    llm=OpenAI(temperature=0),
    toolkit=toolkit,
    verbose=True,
    agent_type=AgentType.ZERO_SHOT_REACT_DESCRIPTION,
)
agent_executor.run(f"我想看数据库里王一的岗位是什么")
# agent_executor.run(f"我想用shell打开我的相机")

改的代代码如下,首先是SQLDatabaseToolkit的定义文件toolkit.py,这里我给这个toolkit里面加入了我自定义的打开相机文件

"""Toolkit for interacting with a SQL database."""
from typing import List

from pydantic import Field

from langchain.agents.agent_toolkits.base import BaseToolkit
from langchain.base_language import BaseLanguageModel
from langchain.sql_database import SQLDatabase
from langchain.tools import BaseTool
from langchain.tools.sql_database.tool import (
    InfoSQLDatabaseTool,
    ListSQLDatabaseTool,
    QueryCheckerTool,
    QuerySQLDataBaseTool,
)
from langchain.tools import BaseTool
from langchain.tools import ShellTool
shell_tool = ShellTool()
class Cam(BaseTool):
    name = "相机"
    description = '使用此工具打开我的相机。'
    def _run(self, expr: str): 
        return shell_tool.run({"commands": "conda activate langchain & python C:/Users/dz/Desktop/camera.py & exit()"})
    def _arun(self, query: str):
        raise NotImplementedError("Async operation not supported yet")
class SQLDatabaseToolkit(BaseToolkit):
    """Toolkit for interacting with SQL databases."""

    db: SQLDatabase = Field(exclude=True)
    llm: BaseLanguageModel = Field(exclude=True)

    @property
    def dialect(self) -> str:
        """Return string representation of dialect to use."""
        return self.db.dialect

    class Config:
        """Configuration for this pydantic object."""

        arbitrary_types_allowed = True

    def get_tools(self) -> List[BaseTool]:
        """Get the tools in the toolkit."""
        query_sql_database_tool_description = (
            "Input to this tool is a detailed and correct SQL query, output is a "
            "result from the database. If the query is not correct, an error message "
            "will be returned. If an error is returned, rewrite the query, check the "
            "query, and try again. If you encounter an issue with Unknown column "
            "'xxxx' in 'field list', using schema_sql_db to query the correct table "
            "fields."
        )
        info_sql_database_tool_description = (
            "Input to this tool is a comma-separated list of tables, output is the "
            "schema and sample rows for those tables. "
            "Be sure that the tables actually exist by calling list_tables_sql_db "
            "first! Example Input: 'table1, table2, table3'"
        )
        return [
            QuerySQLDataBaseTool(
                db=self.db, description=query_sql_database_tool_description
            ),
            InfoSQLDatabaseTool(
                db=self.db, description=info_sql_database_tool_description
            ),
            ListSQLDatabaseTool(db=self.db),
            QueryCheckerTool(db=self.db, llm=self.llm),
            Cam()
        ]

然后是是create_sql_agent里面的SQL_PREFIX的定义文件prompt.py,这里我给让它首先判断改用sell工具还是sql工具

# flake8: noqa

SQL_PREFIX = """You are an agent designed to interact with a SQL database or run your shell.
if you are going to run a shell,please use a tool name is"相机".
Otherwise,given an input question, create a syntactically correct {dialect} query to run, then look at the results of the query and return the answer.
Unless the user specifies a specific number of examples they wish to obtain, always limit your query to at most {top_k} results.
You can order the results by a relevant column to return the most interesting examples in the database.
Never query for all the columns from a specific table, only ask for the relevant columns given the question.
You have access to tools for interacting with the database.
Only use the below tools. Only use the information returned by the below tools to construct your final answer.
You MUST double check your query before executing it. If you get an error while executing a query, rewrite the query and try again.

DO NOT make any DML statements (INSERT, UPDATE, DELETE, DROP etc.) to the database.

If the question does not seem related to the database, just return "I don't know" as the answer.
"""

SQL_SUFFIX = """Begin!

Question: {input}
Thought: I should decide to use which tool,then if I decide not to use '相机' ,then look at the tables in the database to see what I can query.  Then I should query the schema of the most relevant tables.
{agent_scratchpad}"""

4.学习笔记

这里我跟着吴恩达大佬的视频学习,记录一下笔记。

4.1Models,prompts,parsers

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我是小z呀

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值