diff --git a/docs/extras/modules/chains/how_to/learned_prompt_optimization.ipynb b/docs/extras/modules/chains/how_to/learned_prompt_optimization.ipynb new file mode 100644 index 0000000000000..5ff4a95dcc97f --- /dev/null +++ b/docs/extras/modules/chains/how_to/learned_prompt_optimization.ipynb @@ -0,0 +1,830 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Learned prompt variable injection via rl chain\n", + "\n", + "The rl_chain (reinforcement learning chain) is used primarily for prompt variable injection: when we want to enhance a prompt with a value but we are not sure which of the available variable values will make the prompt achieve what we want.\n", + "\n", + "It provides a way to learn a specific prompt engineering policy without fine tuning the underlying foundational model.\n", + "\n", + "The example layed out below is trivial and a strong llm could make a good variable selection and injection without the intervention of this chain, but it is perfect for showcasing the chain's usage. Advanced options and explanations are provided at the end.\n", + "\n", + "The goal of this example scenario is for the chain to select a meal based on the user declared preferences, and inject the meal into the prompt template. The final prompt will then be sent to the llm of choice and the llm output will be returned to the user." + ] + }, + { + "cell_type": "code", + "execution_count": 36, + "metadata": {}, + "outputs": [], + "source": [ + "# four meals defined, some vegetarian some not\n", + "\n", + "meals = [\n", + " \"Beef Enchiladas with Feta cheese. Mexican-Greek fusion\",\n", + " \"Chicken Flatbreads with red sauce. Italian-Mexican fusion\",\n", + " \"Veggie sweet potato quesadillas with vegan cheese\",\n", + " \"One-Pan Tortelonni bake with peppers and onions\",\n", + "]" + ] + }, + { + "cell_type": "code", + "execution_count": 37, + "metadata": {}, + "outputs": [ + { + "data": { + "text/plain": [ + "\"\\n\\nYes, I'm ready.\"" + ] + }, + "execution_count": 37, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "# pick and configure the LLM of your choice\n", + "\n", + "from langchain.llms import OpenAI\n", + "llm = OpenAI(engine=\"text-davinci-003\")\n", + "\n", + "llm.predict(\"are you ready?\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "##### Intialize the rl chain with provided defaults\n", + "\n", + "The prompt template which will be used to query the LLM needs to be defined.\n", + "It can be anything, but here `{meal}` is being used and is going to be replaced by one of the meals above, the rl chain will try to pick and inject the best meal\n" + ] + }, + { + "cell_type": "code", + "execution_count": 38, + "metadata": {}, + "outputs": [], + "source": [ + "from langchain.prompts import PromptTemplate\n", + "\n", + "# here I am using the variable meal which will be replaced by one of the meals above\n", + "# and some variables like user, preference, and text_to_personalize which I will provide at chain run time\n", + "\n", + "PROMPT_TEMPLATE = \"\"\"Here is the description of a meal: \"{meal}\".\n", + "\n", + "Embed the meal into the given text: \"{text_to_personalize}\".\n", + "\n", + "Prepend a personalized message including the user's name \"{user}\" \n", + " and their preference \"{preference}\".\n", + "\n", + "Make it sound good.\n", + "\"\"\"\n", + "\n", + "PROMPT = PromptTemplate(\n", + " input_variables=[\"meal\", \"text_to_personalize\", \"user\", \"preference\"], \n", + " template=PROMPT_TEMPLATE\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Next the rl chain's PickBest chain is being initialized. We must provide the llm of choice and the defined prompt. As the name indicates, the chain's goal is to Pick the Best of the meals that will be provided, based on some criteria. " + ] + }, + { + "cell_type": "code", + "execution_count": 39, + "metadata": {}, + "outputs": [], + "source": [ + "import langchain.chains.rl_chain as rl_chain\n", + "\n", + "chain = rl_chain.PickBest.from_llm(llm=llm, prompt=PROMPT)\n" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Once the chain is setup I am going to call it with the meals I want to be selected from, and some context based on which the chain will select a meal." + ] + }, + { + "cell_type": "code", + "execution_count": 40, + "metadata": {}, + "outputs": [], + "source": [ + "response = chain.run(\n", + " meal = rl_chain.ToSelectFrom(meals),\n", + " user = rl_chain.BasedOn(\"Tom\"),\n", + " preference = rl_chain.BasedOn([\"Vegetarian\", \"regular dairy is ok\"]),\n", + " text_to_personalize = \"This is the weeks specialty dish, our master chefs \\\n", + " believe you will love it!\",\n", + ")" + ] + }, + { + "cell_type": "code", + "execution_count": 41, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Hey Tom! We have an amazing special dish for you this week - veggie sweet potato quesadillas with vegan cheese, which we're sure you'll love as a vegetarian who's ok with regular dairy. Enjoy!\n" + ] + } + ], + "source": [ + "print(response[\"response\"])" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## What is the chain doing\n", + "\n", + "What is happening behind the scenes here is that the rl chain will\n", + "\n", + "- take the meals\n", + "- take the user and their preference\n", + "- based on the user and their preference (context) it will select a meal\n", + "- it will auto-evaluate if that meal selection was good or bad\n", + "- it will finally inject the meal into the prompt and query the llm\n", + "- the user will get the llm response back\n", + "\n", + "Now, the way the chain is doing this is that it is learning a contextual bandit rl model that is trained to make good selections (specifially the [VowpalWabbit](https://github.com/VowpalWabbit/vowpal_wabbit) ML library is being used).\n", + "\n", + "Since this rl model will be untrained when we first start, it might make a random selection that doesn't fit the user and their preferences. But if we give it time to learn the user and their preferences, it should start to make better selections (or quickly learn a good one and just pick that!)." + ] + }, + { + "cell_type": "code", + "execution_count": 42, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\"Hey Tom, our master chefs have prepared something special for you this week - a Mexican-Greek fusion of Beef Enchiladas with Feta cheese that is sure to tantalize your taste buds. Don't worry, we've got you covered with a vegetarian option and regular dairy is ok - so you can enjoy the delicious flavors without any worries!\"\n", + "\n", + "\"Hey Tom! Our master chefs have created a truly unique dish this week, perfect for you! Beef Enchiladas with Feta cheese - a delicious Mexican-Greek fusion - and made with vegetarian ingredients and regular dairy. We know you'll love it!\"\n", + "\n", + "Hey Tom, we have something special for you this week - our veggie sweet potato quesadillas with vegan cheese! We know you like vegetarian dishes and don't mind regular dairy, so we think you'll love this delicious meal.\n", + "\n", + "Hey Tom, we have the perfect dish for you this week! Our master chefs have crafted delicious veggie sweet potato quesadillas with vegan cheese, perfect for vegetarians and those who are okay with regular dairy. We guarantee that you will love it!\n", + "\n", + "Hey Tom! Our master chefs have crafted a delicious Veggie Sweet Potato Quesadillas with vegan cheese, specially designed with your Vegetarian preference in mind - they're sure you will love it! Enjoy this weeks specialty dish!\n", + "\n" + ] + } + ], + "source": [ + "for _ in range(5):\n", + " try:\n", + " response = chain.run(\n", + " meal = rl_chain.ToSelectFrom(meals),\n", + " user = rl_chain.BasedOn(\"Tom\"),\n", + " preference = rl_chain.BasedOn([\"Vegetarian\", \"regular dairy is ok\"]),\n", + " text_to_personalize = \"This is the weeks specialty dish, our master chefs believe you will love it!\",\n", + " )\n", + " except Exception as e:\n", + " print(e)\n", + " print(response[\"response\"])\n", + " print()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## How is the chain learning\n", + "\n", + "The way the chain is learning that Tom prefers veggetarian meals is via an AutoSelectionScorer that is built into the chain. The scorer will call the LLM again and ask it to evaluate the selection (`ToSelectFrom`) using the information wrapped in (`BasedOn`).\n", + "\n", + "You can set `langchain.debug=True` if you want to see the details of the auto-scorer, but you can also define the scoring prompt yourself." + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "metadata": {}, + "outputs": [], + "source": [ + "scoring_criteria_template = \"Given {preference} rank how good or bad this selection is {meal}\"\n", + "\n", + "chain = rl_chain.PickBest.from_llm(\n", + " llm=llm,\n", + " prompt=PROMPT,\n", + " selection_scorer=rl_chain.AutoSelectionScorer(llm=llm, scoring_criteria_template_str=scoring_criteria_template),\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "If you want to examine the score and other selection metadata you can by examining the metadata object returned by the chain" + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\"Hey Tom! We're so excited for you to try out this week's specialty dish. Our master chefs have put together some delicious veggie sweet potato quesadillas with vegan cheese for you, perfect for vegetarians or anyone who's ok with regular dairy. We can't wait for you to enjoy it!\"\n", + "selected index: 2, score: 0.5\n" + ] + } + ], + "source": [ + "response = chain.run(\n", + " meal = rl_chain.ToSelectFrom(meals),\n", + " user = rl_chain.BasedOn(\"Tom\"),\n", + " preference = rl_chain.BasedOn([\"Vegetarian\", \"regular dairy is ok\"]),\n", + " text_to_personalize = \"This is the weeks specialty dish, our master chefs believe you will love it!\",\n", + ")\n", + "print(response[\"response\"])\n", + "selection_metadata = response[\"selection_metadata\"]\n", + "print(f\"selected index: {selection_metadata.selected.index}, score: {selection_metadata.selected.score}\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "In a more realistic scenario it is likely that you have a well defined scoring function for what was selected. For example, you might be doing few-shot prompting and want to select prompt examples for a natural language to sql translation task. In that case the scorer could be: did the sql that was generated run in an sql engine? In that case you want to plugin a scoring function. In the example below I will just check if the meal picked was vegetarian or not." + ] + }, + { + "cell_type": "code", + "execution_count": 10, + "metadata": {}, + "outputs": [], + "source": [ + "class CustomSelectionScorer(rl_chain.SelectionScorer):\n", + " def score_response(\n", + " self, inputs, llm_response: str, event: rl_chain.PickBestEvent) -> float:\n", + "\n", + " print(event.based_on)\n", + " print(event.to_select_from)\n", + "\n", + " # you can build a complex scoring function here\n", + " # it is prefereable that the score ranges between 0 and 1 but it is not enforced\n", + "\n", + " selected_meal = event.to_select_from[\"meal\"][event.selected.index]\n", + " print(f\"selected meal: {selected_meal}\")\n", + "\n", + " if \"Tom\" in event.based_on[\"user\"]:\n", + " if \"Vegetarian\" in event.based_on[\"preference\"]:\n", + " if \"Chicken\" in selected_meal or \"Beef\" in selected_meal:\n", + " return 0.0\n", + " else:\n", + " return 1.0\n", + " else:\n", + " if \"Chicken\" in selected_meal or \"Beef\" in selected_meal:\n", + " return 1.0\n", + " else:\n", + " return 0.0\n", + " else:\n", + " raise NotImplementedError(\"I don't know how to score this user\")" + ] + }, + { + "cell_type": "code", + "execution_count": 11, + "metadata": {}, + "outputs": [], + "source": [ + "chain = rl_chain.PickBest.from_llm(\n", + " llm=llm,\n", + " prompt=PROMPT,\n", + " selection_scorer=CustomSelectionScorer(),\n", + ")" + ] + }, + { + "cell_type": "code", + "execution_count": 12, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "{'user': ['Tom'], 'preference': ['Vegetarian', 'regular dairy is ok']}\n", + "{'meal': ['Beef Enchiladas with Feta cheese. Mexican-Greek fusion', 'Chicken Flatbreads with red sauce. Italian-Mexican fusion', 'Veggie sweet potato quesadillas with vegan cheese', 'One-Pan Tortelonni bake with peppers and onions']}\n", + "selected meal: Veggie sweet potato quesadillas with vegan cheese\n" + ] + } + ], + "source": [ + "response = chain.run(\n", + " meal = rl_chain.ToSelectFrom(meals),\n", + " user = rl_chain.BasedOn(\"Tom\"),\n", + " preference = rl_chain.BasedOn([\"Vegetarian\", \"regular dairy is ok\"]),\n", + " text_to_personalize = \"This is the weeks specialty dish, our master chefs believe you will love it!\",\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## How can I track the chains progress\n", + "\n", + "You can track the chains progress by using the metrics mechanism provided. I am going to expand the users to Tom and Anna, and extend the scoring function. I am going to initialize two chains, one with the default learning policy and one with a built-in random policy (i.e. selects a meal randomly), and plot their scoring progress." + ] + }, + { + "cell_type": "code", + "execution_count": 13, + "metadata": {}, + "outputs": [], + "source": [ + "class CustomSelectionScorer(rl_chain.SelectionScorer):\n", + " def score_preference(self, preference, selected_meal):\n", + " if \"Vegetarian\" in preference:\n", + " if \"Chicken\" in selected_meal or \"Beef\" in selected_meal:\n", + " return 0.0\n", + " else:\n", + " return 1.0\n", + " else:\n", + " if \"Chicken\" in selected_meal or \"Beef\" in selected_meal:\n", + " return 1.0\n", + " else:\n", + " return 0.0\n", + " def score_response(\n", + " self, inputs, llm_response: str, event: rl_chain.PickBestEvent) -> float:\n", + "\n", + " selected_meal = event.to_select_from[\"meal\"][event.selected.index]\n", + "\n", + " if \"Tom\" in event.based_on[\"user\"]:\n", + " return self.score_preference(event.based_on[\"preference\"], selected_meal)\n", + " elif \"Anna\" in event.based_on[\"user\"]:\n", + " return self.score_preference(event.based_on[\"preference\"], selected_meal)\n", + " else:\n", + " raise NotImplementedError(\"I don't know how to score this user\")" + ] + }, + { + "cell_type": "code", + "execution_count": 21, + "metadata": {}, + "outputs": [], + "source": [ + "chain = rl_chain.PickBest.from_llm(\n", + " llm=llm,\n", + " prompt=PROMPT,\n", + " selection_scorer=CustomSelectionScorer(),\n", + " metrics_step=5,\n", + " metrics_window_size=5, # rolling window average\n", + ")\n", + "\n", + "random_chain = rl_chain.PickBest.from_llm(\n", + " llm=llm,\n", + " prompt=PROMPT,\n", + " selection_scorer=CustomSelectionScorer(),\n", + " metrics_step=5,\n", + " metrics_window_size=5, # rolling window average\n", + " policy=rl_chain.PickBestRandomPolicy # set the random policy instead of default\n", + ")" + ] + }, + { + "cell_type": "code", + "execution_count": 22, + "metadata": {}, + "outputs": [], + "source": [ + "for i in range(40):\n", + " try:\n", + " if i % 2:\n", + " chain.run(\n", + " meal = rl_chain.ToSelectFrom(meals),\n", + " user = rl_chain.BasedOn(\"Tom\"),\n", + " preference = rl_chain.BasedOn([\"Vegetarian\", \"regular dairy is ok\"]),\n", + " text_to_personalize = \"This is the weeks specialty dish, our master chefs believe you will love it!\",\n", + " )\n", + " random_chain.run(\n", + " meal = rl_chain.ToSelectFrom(meals),\n", + " user = rl_chain.BasedOn(\"Tom\"),\n", + " preference = rl_chain.BasedOn([\"Vegetarian\", \"regular dairy is ok\"]),\n", + " text_to_personalize = \"This is the weeks specialty dish, our master chefs believe you will love it!\",\n", + " )\n", + " else:\n", + " chain.run(\n", + " meal = rl_chain.ToSelectFrom(meals),\n", + " user = rl_chain.BasedOn(\"Anna\"),\n", + " preference = rl_chain.BasedOn([\"Loves meat\", \"especially beef\"]),\n", + " text_to_personalize = \"This is the weeks specialty dish, our master chefs believe you will love it!\",\n", + " )\n", + " random_chain.run(\n", + " meal = rl_chain.ToSelectFrom(meals),\n", + " user = rl_chain.BasedOn(\"Anna\"),\n", + " preference = rl_chain.BasedOn([\"Loves meat\", \"especially beef\"]),\n", + " text_to_personalize = \"This is the weeks specialty dish, our master chefs believe you will love it!\",\n", + " )\n", + " except Exception as e:\n", + " print(e)" + ] + }, + { + "cell_type": "code", + "execution_count": 25, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "The final average score for the default policy, calculated over a rolling window, is: 1.0\n", + "The final average score for the random policy, calculated over a rolling window, is: 0.4\n" + ] + }, + { + "data": { + "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiMAAAGdCAYAAADAAnMpAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAABmX0lEQVR4nO3dd3RU5drG4d+kh4SEHhIIhN5rgAiIoKIREUWPioB0PBZQEFHBAqhHgaP4oYJgRSw0D4qNIkVAECGELh1Ch4SakARSZvb3x4ZIhEDKJDuZ3NdaWWtmsss9KTPP7P3u57UZhmEgIiIiYhE3qwOIiIhI8aZiRERERCylYkREREQspWJERERELKViRERERCylYkREREQspWJERERELKViRERERCzlYXWA7HA4HBw7doySJUtis9msjiMiIiLZYBgG58+fJyQkBDe3rI9/FIli5NixY4SGhlodQ0RERHLh8OHDVK5cOcvvF4lipGTJkoD5ZAICAixOIyIiItmRkJBAaGhoxvt4VopEMXL51ExAQICKERERkSLmRkMsNIBVRERELKViRERERCylYkREREQsVSTGjGSH3W4nLS3N6hgiLsPd3R0PDw9dTi8i+c4lipHExESOHDmCYRhWRxFxKSVKlCA4OBgvLy+ro4iICyvyxYjdbufIkSOUKFGC8uXL61OciBMYhkFqaionT54kJiaGWrVqXbdhkYhIXhT5YiQtLQ3DMChfvjy+vr5WxxFxGb6+vnh6enLw4EFSU1Px8fGxOpKIuCiX+aijIyIizqejISJSEPRKIyIiIpbKcTGycuVKunTpQkhICDabjXnz5t1wneXLl9O8eXO8vb2pWbMmX3zxRS6iur4OHTowdOjQHK0zb948atasibu7e47XvZ7s/m6vtHz5cmw2G+fOnXNajrzIzXOwyj9/92FhYUycONGyPCIiBSnHxUhSUhJNmjRh8uTJ2Vo+JiaGzp07c+utt7Jp0yaGDh3KwIEDWbRoUY7DytUef/xxHnzwQQ4fPswbb7yRL/s4cOAANpuNTZs25cv288vx48fp1KmT1TFyJSoqin//+99WxxARKRA5HsDaqVOnHL3AT506lWrVqjFhwgQA6tWrx6pVq/i///s/IiMjc7p7uUJiYiJxcXFERkYSEhJidZwCYRgGdrsdD48b/+lWrFixABLlj/Lly1sdQUSkwOT71TRr1qyhY8eOmR6LjIy87imFlJQUUlJSMu4nJCTkVzzLJCUl8eSTT/Ldd99RsmRJhg8fftUyKSkpvPzyy8ycOZNz587RsGFDxo8fT4cOHVi+fDm33norALfddhsAv/32G40aNWLw4MGsXLmSs2fPUqNGDV566SW6d++esd2wsDCGDh2a6XfQtGlTunbtypgxY67KUa1aNQCaNWsGQPv27Vm+fHm2nueqVasYOXIk69evp1y5ctx///2MHTsWPz8/AL766ivee+89du3ahZ+fH7fddhsTJ06kQoUKABnPc/78+bzyyits3bqVX3/9lTFjxtC4cWN8fHz49NNP8fLy4oknnsiU32az8f3339O1a1cOHDhAtWrVmDt3Lh988AFr166lVq1aTJ06ldatW2es88knn/D6669z+vRpIiMjadeuHa+//nqWp54ub3fmzJm8//77bNiwgZo1azJ58mTat2+fsdyKFSt4/vnn2bx5M2XKlKFPnz785z//ybKo+ufv6Ny5c7z44ovMmzeP+Ph4atasybhx47j11lsJDg7m888/58EHH8xYf968efTs2ZMTJ07ccLZMkbw4du4C36w9SHKq3eookkf921YjtEwJS/ad78XIiRMnCAoKyvRYUFAQCQkJXLhw4ZqX444dO5bXXnstV/szDIMLadb8U/h6umf7qp7nn3+eFStW8MMPP1ChQgVeeuklNmzYQNOmTTOWGTx4MNu3b2fWrFmEhITw/fffc9ddd7F161batGnDrl27qFOnDnPnzqVNmzaUKVOGkydPEh4ezosvvkhAQAC//PILvXr1okaNGrRq1SpXz2vdunW0atWKJUuW0KBBg2w3wNq3bx933XUX//nPf/j88885efIkgwcPZvDgwUybNg0wL81+4403qFOnDnFxcQwbNoy+ffsyf/78TNsaMWIE77zzDtWrV6d06dIATJ8+nWHDhrF27VrWrFlD3759adu2LXfccUeWmV5++WXeeecdatWqxcsvv0z37t3Zu3cvHh4erF69mieeeILx48dz7733smTJEl599dVsPdfnn3+eiRMnUr9+fd599126dOlCTEwMZcuW5ejRo9x999307duXL7/8kp07d/LYY4/h4+NzzeLvnxwOB506deL8+fN8/fXX1KhRg+3bt+Pu7o6fnx+PPPII06ZNy1SMXL6vQkTy09mk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19eX2267jbi4OBYsWEC9evUICAigR48eJCcnZ/k8Dx48SJcuXShdujR+fn40aNCA+fPnZ3x/27ZtdOrUCX9/f4KCgujVqxenTp3KcnspKSkMHz6cSpUq4efnR0RExFWzRK9evZoOHTpQokQJSpcuTWRkJGfPnqVv376sWLGC9957D5vNhs1m48CBA9c8TTN37lwaNGiAt7c3YWFhTJgwIdM+wsLCeOutt+jfvz8lS5akSpUqfPzxx1nmhr9/54MHDyYwMJBy5crx6quvZvrdnD17lt69e1O6dGlKlChBp06d2LNnT5bbvNZpms8//zwje3BwMIMHDwagf//+3HPPPZmWTUtLo0KFCnz22WfXzS6ux+4wGDJzIztPnKd8SW8+7dMCP299PhVruF4xYhiQmmTNVw4760+fPh0vLy9Wr17N1KlTAXBzc+P999/nr7/+Yvr06SxbtowXXngh03rJycm88847fPXVV6xcuZJDhw4xfPjwjO9PmDCBL774gs8//5xVq1Zx5swZvv/++0zbeOGFF5g7dy7Tp09nw4YN1KxZk8jISM6cOZNpuTFjxjBp0iT++OMPDh8+zMMPP8zEiROZMWMGv/zyC7/++isffPBBls9x0KBBpKSksHLlSrZu3cr48eMzPsmfO3eO2267jWbNmrF+/XoWLlxIbGwsDz/8cJbbGzx4MGvWrGHWrFls2bKFhx56iLvuuivjDXvTpk3cfvvt1K9fnzVr1rBq1Sq6dOmC3W7nvffeo3Xr1jz22GMcP36c48ePExoaetU+oqOjefjhh3nkkUfYunUrY8aM4dVXX+WLL77ItNyECRNo0aIFGzdu5KmnnuLJJ59k165dWWYH83fu4eHBunXreO+993j33Xf59NNPM77ft29f1q9fz48//siaNWswDIO7776btLS06273silTpjBo0CD+/e9/s3XrVn788Udq1qwJwMCBA1m4cGGmwvXnn38mOTmZbt26ZWv74jrGL9zJ0p1xeHu48UnvFoSU8rU6khRnRhEQHx9vAEZ8fPxV37tw4YKxfft248KFC+YDKYmGMTrAmq+UxGw/p/bt2xvNmjW74XLffvutUbZs2Yz706ZNMwBj7969GY9NnjzZCAoKyrgfHBxs/Pe//824n5aWZlSuXNm47777DMMwjMTERMPT09P45ptvMpZJTU01QkJCMtb77bff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", + "text/plain": [ + "
" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "# note matplotlib is not a dependency of langchain so you need to install to plot\n", + "\n", + "# from matplotlib import pyplot as plt\n", + "# chain.metrics.to_pandas()['score'].plot(label=\"default learning policy\")\n", + "# random_chain.metrics.to_pandas()['score'].plot(label=\"random selection policy\")\n", + "# plt.legend()\n", + "\n", + "print(f\"The final average score for the default policy, calculated over a rolling window, is: {chain.metrics.to_pandas()['score'].iloc[-1]}\")\n", + "print(f\"The final average score for the random policy, calculated over a rolling window, is: {random_chain.metrics.to_pandas()['score'].iloc[-1]}\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "There is a bit of randomness involved in the rl_chain's selection since the chain explores the selection space in order to learn the world as best as it can (see details of default exploration algorithm used [here](https://github.com/VowpalWabbit/vowpal_wabbit/wiki/Contextual-Bandit-Exploration-with-SquareCB)), but overall, default chain policy should be doing better than random as it learns" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Advanced options\n", + "\n", + "The rl chain is highly configurable in order to be able to adjust to various selection scenarios. If you want to learn more about the ML library that powers it please take a look at tutorials [here](https://vowpalwabbit.org/)\n" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "| Section | Description | Example / Usage |\n", + "|---------|-------------|-----------------|\n", + "| [**Change Chain Logging Level**](#change-chain-logging-level) | Change the logging level for the RL chain. | `logger.setLevel(logging.INFO)` |\n", + "| [**Featurization**](#featurization) | Adjusts the input to the RL chain. Can set auto-embeddings ON for more complex embeddings. | `chain = rl_chain.PickBest.from_llm(auto_embed=True, [...])` |\n", + "| [**Learned Policy to Learn Asynchronously**](#learned-policy-to-learn-asynchronously) | Score asynchronously if user input is needed for scoring. | `chain.update_with_delayed_score(score=, chain_response=response)` |\n", + "| [**Store Progress of Learned Policy**](#store-progress-of-learned-policy) | Option to store the progress of the variable injection learned policy. | `chain.save_progress()` |\n", + "| [**Stop Learning of Learned Policy**](#stop-learning-of-learned-policy) | Toggle the RL chain's learned policy updates ON/OFF. | `chain.deactivate_selection_scorer()` |\n", + "| [**Set a Different Policy**](#set-a-different-policy) | Choose between different policies: default, random, or custom. | Custom policy creation at chain creation time. |\n", + "| [**Different Exploration Algorithms and Options for Default Learned Policy**](#different-exploration-algorithms-and-options-for-the-default-learned-policy) | Set different exploration algorithms and hyperparameters for `VwPolicy`. | `vw_cmd = [\"--cb_explore_adf\", \"--quiet\", \"--squarecb\", \"--interactions=::\"]` |\n", + "| [**Learn Policy's Data Logs**](#learned-policys-data-logs) | Store and examine `VwPolicy`'s data logs. | `chain = rl_chain.PickBest.from_llm(vw_logs=, [...])` |\n", + "| [**Other Advanced Featurization Options**](#other-advanced-featurization-options) | Specify advanced featurization options for the RL chain. | `age = rl_chain.BasedOn(\"age:32\")` |\n", + "| [**More Info on Auto or Custom SelectionScorer**](#more-info-on-auto-or-custom-selectionscorer) | Dive deeper into how selection scoring is determined. | `selection_scorer=rl_chain.AutoSelectionScorer(llm=llm, scoring_criteria_template_str=scoring_criteria_template)` |\n", + "\n", + "### change chain logging level\n", + "\n", + "```\n", + "import logging\n", + "logger = logging.getLogger(\"rl_chain\")\n", + "logger.setLevel(logging.INFO)\n", + "```\n", + "\n", + "### featurization\n", + "\n", + "#### auto_embed\n", + "\n", + "By default the input to the rl chain (`ToSelectFrom`, `BasedOn`) is not tampered with. This might not be sufficient featurization, so based on how complex the scenario is you can set auto-embeddings to ON\n", + "\n", + "`chain = rl_chain.PickBest.from_llm(auto_embed=True, [...])`\n", + "\n", + "This will produce more complex embeddings and featurizations of the inputs, likely accelerating RL chain learning, albeit at the cost of increased runtime.\n", + "\n", + "By default, [sbert.net's sentence_transformers's ](https://www.sbert.net/docs/pretrained_models.html#model-overview) `all-mpnet-base-v2` model will be used for these embeddings but you can set a different embeddings model by initializing the chain with it as shown in this example. You could also set an entirely different embeddings encoding object, as long as it has an `encode()` function that returns a list of the encodings.\n", + "\n", + "```\n", + "from sentence_transformers import SentenceTransformer\n", + "\n", + "chain = rl_chain.PickBest.from_llm(\n", + " [...]\n", + " feature_embedder=rl_chain.PickBestFeatureEmbedder(\n", + " auto_embed=True,\n", + " model=SentenceTransformer(\"all-mpnet-base-v2\")\n", + " )\n", + ")\n", + "```\n", + "\n", + "#### explicitly defined embeddings\n", + "\n", + "Another option is to define what inputs you think should be embedded manually:\n", + "- `auto_embed = False`\n", + "- Can wrap individual variables in `rl_chain.Embed()` or `rl_chain.EmbedAndKeep()` e.g. `user = rl_chain.BasedOn(rl_chain.Embed(\"Tom\"))`\n", + "\n", + "#### custom featurization\n", + "\n", + "Another final option is to define and set a custom featurization/embedder class that returns a valid input for the learned policy.\n", + "\n", + "## learned policy to learn asynchronously\n", + "\n", + "If to score the result you need input from the user (e.g. my application showed Tom the selected meal and Tom clicked on it, but Anna did not), then the scoring can be done asynchronously. The way to do that is:\n", + "\n", + "- set `selection_scorer=None` on the chain creation OR call `chain.deactivate_selection_scorer()`\n", + "- call the chain for a specific input\n", + "- keep the chain's response (`response = chain.run([...])`)\n", + "- once you have determined the score of the response/chain selection call the chain with it: `chain.update_with_delayed_score(score=, chain_response=response)`\n", + "\n", + "### store progress of learned policy\n", + "\n", + "Since the variable injection learned policy evolves over time, there is the option to store its progress and continue learning. This can be done by calling:\n", + "\n", + "`chain.save_progress()`\n", + "\n", + "which will store the rl chain's learned policy in a file called `latest.vw`. It will also store it in a file with a timestamp. That way, if `save_progress()` is called more than once, multiple checkpoints will be created, but the latest one will always be in `latest.vw`\n", + "\n", + "Next time the chain is loaded, the chain will look for a file called `latest.vw` and if the file exists it will be loaded into the chain and the learning will continue from there.\n", + "\n", + "By default the rl chain model checkpoints will be stored in the current directory but you can specify the save/load location at chain creation time:\n", + "\n", + "`chain = rl_chain.PickBest.from_llm(model_save_dir=, [...])`\n", + "\n", + "### stop learning of learned policy\n", + "\n", + "If you want the rl chain's learned policy to stop updating you can turn it off/on:\n", + "\n", + "`chain.deactivate_selection_scorer()` and `chain.activate_selection_scorer()`\n", + "\n", + "### set a different policy\n", + "\n", + "There are two policies currently available:\n", + "\n", + "- default policy: `VwPolicy` which learns a [Vowpal Wabbit](https://github.com/VowpalWabbit/vowpal_wabbit) [Contextual Bandit](https://github.com/VowpalWabbit/vowpal_wabbit/wiki/Contextual-Bandit-algorithms) model\n", + "\n", + "- random policy: `RandomPolicy` which doesn't learn anything and just selects a value randomly. this policy can be used to compare other policies with a random baseline one.\n", + "\n", + "- custom policies: a custom policy could be created and set at chain creation time\n", + "\n", + "### different exploration algorithms and options for the default learned policy\n", + "\n", + "The default `VwPolicy` is initialized with some default arguments. The default exploration algorithm is [SquareCB](https://github.com/VowpalWabbit/vowpal_wabbit/wiki/Contextual-Bandit-Exploration-with-SquareCB) but other Contextual Bandit exploration algorithms can be set, and other hyper parameters can be tuned (see [here](https://vowpalwabbit.org/docs/vowpal_wabbit/python/9.6.0/command_line_args.html) for available options).\n", + "\n", + "`vw_cmd = [\"--cb_explore_adf\", \"--quiet\", \"--squarecb\", \"--interactions=::\"]`\n", + "\n", + "`chain = rl_chain.PickBest.from_llm(vw_cmd = vw_cmd, [...])`\n", + "\n", + "### learned policy's data logs\n", + "\n", + "The `VwPolicy`'s data files can be stored and examined or used to do [off policy evaluation](https://vowpalwabbit.org/docs/vowpal_wabbit/python/latest/tutorials/off_policy_evaluation.html) for hyper parameter tuning.\n", + "\n", + "The way to do this is to set a log file path to `vw_logs` on chain creation:\n", + "\n", + "`chain = rl_chain.PickBest.from_llm(vw_logs=, [...])`\n", + "\n", + "### other advanced featurization options\n", + "\n", + "Explictly numerical features can be provided with a colon separator:\n", + "`age = rl_chain.BasedOn(\"age:32\")`\n", + "\n", + "`ToSelectFrom` can be a bit more complex if the scenario demands it, instead of being a list of strings it can be:\n", + "- a list of list of strings:\n", + " ```\n", + " meal = rl_chain.ToSelectFrom([\n", + " [\"meal 1 name\", \"meal 1 description\"],\n", + " [\"meal 2 name\", \"meal 2 description\"]\n", + " ])\n", + " ```\n", + "- a list of dictionaries:\n", + " ```\n", + " meal = rl_chain.ToSelectFrom([\n", + " {\"name\":\"meal 1 name\", \"description\" : \"meal 1 description\"},\n", + " {\"name\":\"meal 2 name\", \"description\" : \"meal 2 description\"}\n", + " ])\n", + " ```\n", + "- a list of dictionaries containing lists:\n", + " ```\n", + " meal = rl_chain.ToSelectFrom([\n", + " {\"name\":[\"meal 1\", \"complex name\"], \"description\" : \"meal 1 description\"},\n", + " {\"name\":[\"meal 2\", \"complex name\"], \"description\" : \"meal 2 description\"}\n", + " ])\n", + " ```\n", + "\n", + "`BasedOn` can also take a list of strings:\n", + "```\n", + "user = rl_chain.BasedOn([\"Tom Joe\", \"age:32\", \"state of california\"])\n", + "```\n", + "\n", + "there is no dictionary provided since multiple variables can be supplied wrapped in `BasedOn`\n", + "\n", + "Storing the data logs into a file allows the examination of what different inputs do to the data format.\n", + "\n", + "### More info on Auto or Custom SelectionScorer\n", + "\n", + "It is very important to get the selection scorer right since the policy uses it to learn. It determines what is called the reward in reinforcement learning, and more specifically in our Contextual Bandits setting.\n", + "\n", + "The general advice is to keep the score between [0, 1], 0 being the worst selection, 1 being the best selection from the available `ToSelectFrom` variables, based on the `BasedOn` variables, but should be adjusted if the need arises.\n", + "\n", + "In the examples provided above, the AutoSelectionScorer is set mostly to get users started but in real world scenarios it will most likely not be an adequate scorer function.\n", + "\n", + "The example also provided the option to change part of the scoring prompt template that the AutoSelectionScorer used to determine whether a selection was good or not:\n", + "\n", + "```\n", + "scoring_criteria_template = \"Given {preference} rank how good or bad this selection is {meal}\"\n", + "chain = rl_chain.PickBest.from_llm(\n", + " llm=llm,\n", + " prompt=PROMPT,\n", + " selection_scorer=rl_chain.AutoSelectionScorer(llm=llm, scoring_criteria_template_str=scoring_criteria_template),\n", + ")\n", + "\n", + "```\n", + "\n", + "Internally the AutoSelectionScorer adjusted the scoring prompt to make sure that the llm scoring retured a single float.\n", + "\n", + "However, if needed, a FULL scoring prompt can also be provided:\n" + ] + }, + { + "cell_type": "code", + "execution_count": 17, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "\u001b[32;1m\u001b[1;3m[chain/start]\u001b[0m \u001b[1m[1:chain:PickBest] Entering Chain run with input:\n", + "\u001b[0m[inputs]\n", + "\u001b[32;1m\u001b[1;3m[chain/start]\u001b[0m \u001b[1m[1:chain:PickBest > 2:chain:LLMChain] Entering Chain run with input:\n", + "\u001b[0m[inputs]\n", + "\u001b[32;1m\u001b[1;3m[llm/start]\u001b[0m \u001b[1m[1:chain:PickBest > 2:chain:LLMChain > 3:llm:OpenAI] Entering LLM run with input:\n", + "\u001b[0m{\n", + " \"prompts\": [\n", + " \"Here is the description of a meal: \\\"Beef Enchiladas with Feta cheese. Mexican-Greek fusion\\\".\\n\\nEmbed the meal into the given text: \\\"This is the weeks specialty dish, our master chefs believe you will love it!\\\".\\n\\nPrepend a personalized message including the user's name Tom and their preference ['Vegetarian', 'regular dairy is ok'].\\n\\nMake it sound good.\"\n", + " ]\n", + "}\n", + "\u001b[36;1m\u001b[1;3m[llm/end]\u001b[0m \u001b[1m[1:chain:PickBest > 2:chain:LLMChain > 3:llm:OpenAI] [1.63s] Exiting LLM run with output:\n", + "\u001b[0m{\n", + " \"generations\": [\n", + " [\n", + " {\n", + " \"text\": \"\\nHey Tom, we have a special treat this week! Our master chefs have created a Mexican-Greek fusion dish of Beef Enchiladas with Feta cheese - perfect for those who enjoy vegetarian options and can enjoy regular dairy. We know you're going to love it!\",\n", + " \"generation_info\": {\n", + " \"finish_reason\": \"stop\",\n", + " \"logprobs\": null\n", + " }\n", + " }\n", + " ]\n", + " ],\n", + " \"llm_output\": {\n", + " \"token_usage\": {\n", + " \"prompt_tokens\": 89,\n", + " \"total_tokens\": 145,\n", + " \"completion_tokens\": 56\n", + " },\n", + " \"model_name\": \"text-davinci-003\"\n", + " },\n", + " \"run\": null\n", + "}\n", + "\u001b[36;1m\u001b[1;3m[chain/end]\u001b[0m \u001b[1m[1:chain:PickBest > 2:chain:LLMChain] [1.63s] Exiting Chain run with output:\n", + "\u001b[0m{\n", + " \"text\": \"\\nHey Tom, we have a special treat this week! Our master chefs have created a Mexican-Greek fusion dish of Beef Enchiladas with Feta cheese - perfect for those who enjoy vegetarian options and can enjoy regular dairy. We know you're going to love it!\"\n", + "}\n", + "\u001b[32;1m\u001b[1;3m[chain/start]\u001b[0m \u001b[1m[1:chain:LLMChain] Entering Chain run with input:\n", + "\u001b[0m[inputs]\n", + "\u001b[32;1m\u001b[1;3m[llm/start]\u001b[0m \u001b[1m[1:chain:LLMChain > 2:llm:OpenAI] Entering LLM run with input:\n", + "\u001b[0m{\n", + " \"prompts\": [\n", + " \"Given ['Vegetarian', 'regular dairy is ok'] rank how good or bad this selection is ['Beef Enchiladas with Feta cheese. Mexican-Greek fusion', 'Chicken Flatbreads with red sauce. Italian-Mexican fusion', 'Veggie sweet potato quesadillas with vegan cheese', 'One-Pan Tortelonni bake with peppers and onions'], IMPORANT: you MUST return a single number between -1 and 1, -1 being bad, 1 being good\"\n", + " ]\n", + "}\n", + "\u001b[36;1m\u001b[1;3m[llm/end]\u001b[0m \u001b[1m[1:chain:LLMChain > 2:llm:OpenAI] [487ms] Exiting LLM run with output:\n", + "\u001b[0m{\n", + " \"generations\": [\n", + " [\n", + " {\n", + " \"text\": \"\\n\\n0.5\",\n", + " \"generation_info\": {\n", + " \"finish_reason\": \"stop\",\n", + " \"logprobs\": null\n", + " }\n", + " }\n", + " ]\n", + " ],\n", + " \"llm_output\": {\n", + " \"token_usage\": {\n", + " \"prompt_tokens\": 104,\n", + " \"total_tokens\": 109,\n", + " \"completion_tokens\": 5\n", + " },\n", + " \"model_name\": \"text-davinci-003\"\n", + " },\n", + " \"run\": null\n", + "}\n", + "\u001b[36;1m\u001b[1;3m[chain/end]\u001b[0m \u001b[1m[1:chain:LLMChain] [488ms] Exiting Chain run with output:\n", + "\u001b[0m{\n", + " \"text\": \"\\n\\n0.5\"\n", + "}\n", + "\u001b[36;1m\u001b[1;3m[chain/end]\u001b[0m \u001b[1m[1:chain:PickBest] [2.13s] Exiting Chain run with output:\n", + "\u001b[0m[outputs]\n" + ] + }, + { + "data": { + "text/plain": [ + "{'response': \"Hey Tom, we have a special treat this week! Our master chefs have created a Mexican-Greek fusion dish of Beef Enchiladas with Feta cheese - perfect for those who enjoy vegetarian options and can enjoy regular dairy. We know you're going to love it!\",\n", + " 'selection_metadata': }" + ] + }, + "execution_count": 17, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "from langchain.prompts.prompt import PromptTemplate\n", + "import langchain\n", + "langchain.debug = True\n", + "\n", + "REWARD_PROMPT_TEMPLATE = \"\"\"\n", + "\n", + "Given {preference} rank how good or bad this selection is {meal}\n", + "\n", + "IMPORANT: you MUST return a single number between -1 and 1, -1 being bad, 1 being good\n", + "\n", + "\"\"\"\n", + "\n", + "\n", + "REWARD_PROMPT = PromptTemplate(\n", + " input_variables=[\"preference\", \"meal\"],\n", + " template=REWARD_PROMPT_TEMPLATE,\n", + ")\n", + "\n", + "chain = rl_chain.PickBest.from_llm(\n", + " llm=llm,\n", + " prompt=PROMPT,\n", + " selection_scorer=rl_chain.AutoSelectionScorer(llm=llm, prompt=REWARD_PROMPT),\n", + ")\n", + "\n", + "chain.run(\n", + " meal = rl_chain.ToSelectFrom(meals),\n", + " user = rl_chain.BasedOn(\"Tom\"),\n", + " preference = rl_chain.BasedOn([\"Vegetarian\", \"regular dairy is ok\"]),\n", + " text_to_personalize = \"This is the weeks specialty dish, our master chefs believe you will love it!\",\n", + ")" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "venv", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.8.16" + }, + "orig_nbformat": 4 + }, + "nbformat": 4, + "nbformat_minor": 2 +} diff --git a/libs/langchain/langchain/chains/rl_chain/__init__.py b/libs/langchain/langchain/chains/rl_chain/__init__.py index 80242139f5c1e..f112dcea092fb 100644 --- a/libs/langchain/langchain/chains/rl_chain/__init__.py +++ b/libs/langchain/langchain/chains/rl_chain/__init__.py @@ -16,6 +16,7 @@ PickBest, PickBestEvent, PickBestFeatureEmbedder, + PickBestRandomPolicy, PickBestSelected, ) @@ -39,6 +40,7 @@ def configure_logger() -> None: "PickBestEvent", "PickBestSelected", "PickBestFeatureEmbedder", + "PickBestRandomPolicy", "Embed", "BasedOn", "ToSelectFrom", diff --git a/libs/langchain/langchain/chains/rl_chain/base.py b/libs/langchain/langchain/chains/rl_chain/base.py index 4b5ac572f9c6d..26ac9a43e132c 100644 --- a/libs/langchain/langchain/chains/rl_chain/base.py +++ b/libs/langchain/langchain/chains/rl_chain/base.py @@ -166,7 +166,7 @@ def __init__(self, inputs: Dict[str, Any], selected: Optional[TSelected] = None) TEvent = TypeVar("TEvent", bound=Event) -class Policy(ABC): +class Policy(Generic[TEvent], ABC): def __init__(self, **kwargs: Any): pass diff --git a/libs/langchain/langchain/chains/rl_chain/pick_best_chain.py b/libs/langchain/langchain/chains/rl_chain/pick_best_chain.py index 791d12cdb4622..0da0780313cb4 100644 --- a/libs/langchain/langchain/chains/rl_chain/pick_best_chain.py +++ b/libs/langchain/langchain/chains/rl_chain/pick_best_chain.py @@ -223,6 +223,21 @@ def format(self, event: PickBestEvent) -> str: return self.format_auto_embed_off(event) +class PickBestRandomPolicy(base.Policy[PickBestEvent]): + def __init__(self, feature_embedder: base.Embedder, **kwargs: Any): + self.feature_embedder = feature_embedder + + def predict(self, event: PickBestEvent) -> List[Tuple[int, float]]: + num_items = len(event.to_select_from) + return [(i, 1.0 / num_items) for i in range(num_items)] + + def learn(self, event: PickBestEvent) -> None: + pass + + def log(self, event: PickBestEvent) -> None: + pass + + class PickBest(base.RLChain[PickBestEvent]): """ `PickBest` is a class designed to leverage the Vowpal Wabbit (VW) model for reinforcement learning with a context, with the goal of modifying the prompt before the LLM call. diff --git a/libs/langchain/poetry.lock b/libs/langchain/poetry.lock index 067524c6cec58..75aeba8d087a1 100644 --- a/libs/langchain/poetry.lock +++ b/libs/langchain/poetry.lock @@ -10939,7 +10939,7 @@ cffi = {version = ">=1.11", markers = "platform_python_implementation == \"PyPy\ cffi = ["cffi (>=1.11)"] [extras] -all = ["O365", "aleph-alpha-client", "amadeus", "arxiv", "atlassian-python-api", "awadb", "azure-ai-formrecognizer", "azure-ai-vision", "azure-cognitiveservices-speech", "azure-cosmos", "azure-identity", "beautifulsoup4", "clarifai", "clickhouse-connect", "cohere", "deeplake", "docarray", "duckduckgo-search", "elasticsearch", "esprima", "faiss-cpu", "google-api-python-client", "google-auth", "google-search-results", "gptcache", "html2text", "huggingface_hub", "jinja2", "jq", "lancedb", "langkit", "lark", "libdeeplake", "librosa", "lxml", "manifest-ml", "marqo", "momento", "nebula3-python", "neo4j", "networkx", "nlpcloud", "nltk", "nomic", "openai", "openlm", "opensearch-py", "pdfminer-six", "pexpect", "pgvector", "pinecone-client", "pinecone-text", "psycopg2-binary", "pymongo", "pyowm", "pypdf", "pytesseract", "python-arango", "pyvespa", "qdrant-client", "rdflib", "redis", "requests-toolbelt", "sentence-transformers", "singlestoredb", "tensorflow-text", "tigrisdb", "tiktoken", "torch", "transformers", "weaviate-client", "wikipedia", "wolframalpha"] +all = ["O365", "aleph-alpha-client", "amadeus", "arxiv", "atlassian-python-api", "awadb", "azure-ai-formrecognizer", "azure-ai-vision", "azure-cognitiveservices-speech", "azure-cosmos", "azure-identity", "beautifulsoup4", "clarifai", "clickhouse-connect", "cohere", "deeplake", "docarray", "duckduckgo-search", "elasticsearch", "esprima", "faiss-cpu", "google-api-python-client", "google-auth", "google-search-results", "gptcache", "html2text", "huggingface_hub", "jinja2", "jq", "lancedb", "langkit", "lark", "libdeeplake", "librosa", "lxml", "manifest-ml", "marqo", "momento", "nebula3-python", "neo4j", "networkx", "nlpcloud", "nltk", "nomic", "openai", "openlm", "opensearch-py", "pdfminer-six", "pexpect", "pgvector", "pinecone-client", "pinecone-text", "psycopg2-binary", "pymongo", "pyowm", "pypdf", "pytesseract", "python-arango", "pyvespa", "qdrant-client", "rdflib", "redis", "requests-toolbelt", "sentence-transformers", "singlestoredb", "tensorflow-text", "tigrisdb", "tiktoken", "torch", "transformers", "vowpal-wabbit-next", "weaviate-client", "wikipedia", "wolframalpha"] azure = ["azure-ai-formrecognizer", "azure-ai-vision", "azure-cognitiveservices-speech", "azure-core", "azure-cosmos", "azure-identity", "azure-search-documents", "openai"] clarifai = ["clarifai"] cohere = ["cohere"] @@ -10955,4 +10955,4 @@ text-helpers = ["chardet"] [metadata] lock-version = "2.0" python-versions = ">=3.8.1,<4.0" -content-hash = "71842b0ce1bd5c663e96a8ef14f71ce42667833cab72de4273ca07241c4465a9" +content-hash = "7bffde1b8d57bad4b5a48d73250cb8276eb7e40dfe19f8490d5f4a25cb15322d" diff --git a/libs/langchain/pyproject.toml b/libs/langchain/pyproject.toml index 00b90087efa9d..198836e41851c 100644 --- a/libs/langchain/pyproject.toml +++ b/libs/langchain/pyproject.toml @@ -295,6 +295,7 @@ all = [ "amadeus", "librosa", "python-arango", + "vowpal-wabbit-next", ] # An extra used to be able to add extended testing.