* Update model kwargs and temperature values
* Update keyboard shortcuts for advanced editing
* make Message field have no handles
* Update OpenAI API Key handling in OpenAIEmbeddingsComponent
* Remove unnecessary field_type key from CustomComponent class
* Update required field behavior in CustomComponent class
* Refactor AzureOpenAIModel.py: Removed unnecessary "required" attribute from input parameters
* Update BaiduQianfanChatModel and OpenAIModel configurations
* Fix range_spec step type validation
* Update RangeSpec step_type default value to "float"
* Fix Save debounce
* Update parameterUtils to use debounce instead of throttle
* Update input type options in schemas and graph base classes
* Refactor run_flow_with_caching endpoint to include simplified and experimental versions
* Add PythonFunctionComponent and test case for it
* Add nest_asyncio to fix event loop issue
* Refactor test_initial_setup.py to use RunOutputs instead of ResultData
* Remove unused code in test_endpoints.py
* Add asyncio loop to uvicorn command
* Refactor load_session method to handle coroutine result
* Fixed saving
* Fixed debouncing
* Add InputType and OutputType literals to schema.py
* Update input type in Graph class
* Add new schema for simplified API request
* Add delete_messages function and update test_successful_run assertions
* Add STREAM_INFO_TEXT constant to model components
* Add session_id to simplified_run_flow_with_caching endpoint
* Add field_typing import to OpenAIModel.py
* update starter projects
* Add constants for Langflow base module
* Update setup.py to include latest component versions
* Update Starter Examples
* sets starter_project fixture to Basic Prompting
* Refactor test_endpoints.py: Update test names and add new tests for different output types
* Update HuggingFace Spaces link and add image for dark mode
* Remove filepath reference
* Update Vertex params in base.py
* Add tests for different input types
* Add type annotations and improve test coverage
* Add duplicate space link to README
* Update HuggingFace Spaces badge in README
* Add Python 3.10 installation requirement to README
* Refactor flow running endpoints
* Refactor SimplifiedAPIRequest and add documentation for Tweaks
* Refactor input_request parameter in simplified_run_flow function
* Add support for retrieving specific component output
* Add custom Uvicorn worker for Langflow application
* Add asyncio loop to LangflowApplication initialization
* Update Makefile with new variables and start command
* Fix indentation in Makefile
* Refactor run_graph function to add support for running a JSON flow
* Refactor getChatInputField function and update API code
* Update HuggingFace Spaces documentation with duplication process
* Add asyncio event loop to uvicorn command
* Add installation of backend in start target
* udpate some starter projects
* Fix formatting in hugging-face-spaces.mdx
* Update installation instructions for Langflow
* set examples order
* Update start command in Makefile
* Add installation and usage instructions for Langflow
* Update Langflow installation and usage instructions
* Fix langflow command in README.md
* Fix broken link to HuggingFace Spaces guide
* Add new SVG assets for blog post, chat bot, and cloud docs
* Refactor example rendering in NewFlowModal
* Add new SVG file for short bio section
* Remove unused import and add new component
* Update title in usage.mdx
* Update HuggingFace Spaces heading in usage.mdx
* Update usage instructions in getting-started/usage.mdx
* Update cache option in usage documentation
* Remove 'advanced' flag from 'n_messages' parameter in MemoryComponent.py
* Refactor code to improve performance and readability
* Update project names and flow examples
* fix document qa example
* Remove commented out code in sidebars.js
* Delete unused documentation files
* Fix bug in login functionality
* Remove global variables from components
* Fix bug in login functionality
* fix modal returning to input
* Update max-width of chat message sender name
* Update styling for chat message component
* Refactor OpenAIEmbeddingsComponent signature
* Update usage.mdx file
* Update path in Makefile
* Add new migration and what's new documentation files
* Add new chapters and migration guides
* Update version to 0.0.13 in pyproject.toml
* new locks
* Update dependencies in pyproject.toml
* general fixes
* Update dependencies in pyproject.toml and poetry.lock files
* add padding to modal
* ✨ (undrawCards/index.tsx): update the SVG used for BasicPrompt component to undraw_short_bio_re_fmx0.svg to match the desired design
♻️ (undrawCards/index.tsx): adjust the width and height of the BasicPrompt SVG to 65% to improve the visual appearance
* Commented out components/data in sidebars.js
* Refactor component names in outputs.mdx
* Update embedded chat script URL
* Add data component and fix formatting in outputs component
* Update dependencies in poetry.lock and pyproject.toml
* Update dependencies in poetry.lock and pyproject.toml
* Refactor code to improve performance and readability
* Update dependencies in poetry.lock and pyproject.toml
* Fixed IO Modal updates
* Remove dead code at API Modal
* Fixed overflow at CodeTabsComponent tweaks page
* ✨ (NewFlowModal/index.tsx): update the name of the example from "Blog Writter" to "Blog Writer" for better consistency and clarity
* Update dependencies versions
* Update langflow-base to version 0.0.15 and fix setup_env script
* Update dependencies in pyproject.toml
* Lock dependencies in parallel
* Add logging statement to setup_app function
* Fix Ace not having type="module" and breaking build
* Update authentication settings for access token cookie
* Update package versions in package-lock.json
* Add scripts directory to Dockerfile
* Add setup_env command to build_and_run target
* Remove unnecessary make command in setup_env
* Remove unnecessary installation step in build_and_run
* Add debug configuration for CLI
* 🔧 chore(Makefile): refactor build_langflow target to use a separate script for updating dependencies and building
✨ feat(update_dependencies.py): add script to update pyproject.toml dependency version based on langflow-base version in src/backend/base/pyproject.toml
* Add number_of_results parameter to AstraDBSearchComponent
* Update HuggingFace Spaces links
* Remove duplicate imports in hugging-face-spaces.mdx
* Add number_of_results parameter to vector search components
* Fixed supabase not commited
* Revert "Fixed supabase not commited"
This reverts commit afb10a6262.
* Update duplicate-space.png image
* Delete unused files and components
* Add/update script to update dependencies
* Add .bak files to .gitignore
* Update version numbers and remove unnecessary dependencies
* Update langflow-base dependency path
* Add Text import to VertexAiModel.py
* Update langflow-base version to 0.0.16 and update dependencies
* Delete start projects and commit session in delete_start_projects function
* Refactor backend startup script to handle autologin option
* Update poetry installation script to include pipx update check
* Update pipx installation script for different operating systems
* Update Makefile to improve setup process
* Add error handling on streaming and fix streaming bug on error
* Added description to Blog Writer
* Sort base classes alphabetically
* Update duplicate-space.png image
* update position on langflow prompt chaining
* Add Langflow CLI and first steps documentation
* Add exception handling for missing 'content' field in search_with_vector_store method
* Remove unused import and update type hinting
* fix bug on egdes after creating group component
* Refactor APIRequest class and update model imports
* Remove unused imports and fix formatting issues
* Refactor reactflowUtils and styleUtils
* Add CLI documentation to getting-started/cli.mdx
* Add CLI usage instructions
* Add ZoomableImage component to first-steps.mdx
* Update CLI and first steps documentation
* Remove duplicate import and add new imports for ThemedImage and useBaseUrl
* Update Langflow CLI documentation link
* Remove first-steps.mdx and update index.mdx and sidebars.js
* Update Docusaurus dependencies
* Add AstraDB RAG Flow guide
* Remove unused imports
* Remove unnecessary import statement
* Refactor guide for better readability
* Add data component documentation
* Update component headings and add prompt template
* Fix logging level and version display
* Add datetime import and buffer for alembic log
* Update flow names in NewFlowModal and documentation
* Add starter projects to sidebars.js
* Fix error handling in DirectoryReader class
* Handle exception when loading components in setup.py
* Update version numbers in pyproject.toml files
* Update build_langflow_base and build_langflow_backup in Makefile
* Added docs
* Update dependencies and build process
* Add Admonition component for API Key documentation
* Update API endpoint in async-api.mdx
* Remove async-api guidelines
* Fix UnicodeDecodeError in DirectoryReader
* Update dependency version and fix encoding issues
* Add conditional build and publish for base and main projects
* Update version to 1.0.0a2 in pyproject.toml
* Remove duplicate imports and unnecessary code in custom-component.mdx
* Fix poetry lock command in Makefile
* Update package versions in pyproject.toml
* Remove unused components and update imports
* 📦 chore(pre-release-base.yml): add pre-release workflow for base project
📦 chore(pre-release-langflow.yml): add pre-release workflow for langflow project
* Add ChatLiteLLMModelComponent to models package
* Add frontend installation and build steps
* Add Dockerfile for building and pushing base image
* Add emoji package and nest-asyncio dependency
* 📝 (components.mdx): update margin style of ZoomableImage to improve spacing
📝 (features.mdx): update margin style of ZoomableImage to improve spacing
📝 (login.mdx): update margin style of ZoomableImage to improve spacing
* Fix module import error in validate.py
* Fix error message in directory_reader.py
* Update version import and handle ImportError
* Add cryptography and langchain-openai dependencies
* Update poetry installation and remove poetry-monorepo-dependency-plugin
* Update workflow and Dockerfile for Langflow base pre-release
* Update display names and descriptions for AstraDB components
* Update installation instructions for Langflow
* Update Astra DB links and remove unnecessary imports
* Rename AstraDB
* Add new components and images
* Update HuggingFace Spaces URLs
* Update Langflow documentation and add new starter projects
* Update flow name to "Basic Prompting (Hello, world!)" in relevant files
* Update Basic Prompting flow name to "Ahoy World!"
* Remove HuggingFace Spaces documentation
* Add new files and update sidebars.js
* Remove async-tasks.mdx and update sidebars.js
* Update starter project URLs
* Enable migration of global variables
* Update OpenAIEmbeddings deployment and model
* 📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (inputs.mdx): add margin to image style to improve spacing and center alignment
📝 (rag-with-astradb.mdx): add margin to image styles to improve spacing and readability
* Update welcome message in index.mdx
* Add global variable feature to Langflow documentation
* Reorganized sidebar categories
* Update migration documentation
* Refactor SplitTextComponent class to accept inputs of type Record and Text
* Adjust embeddings docs
* ✨ (cardComponent/index.tsx): add a minimum height to the card component to ensure consistent layout and prevent content from overlapping when the card is empty or has minimal content
* Update flow name from "Ahoy World!" to "Hello, world!"
* Update documentation for embeddings, models, and vector stores
* Update CreateRecordComponent and parameterUtils.ts
* Add documentation for Text and Record types
* Remove commented lines in sidebars.js
* Add run_flow_from_json function to load.py
* Update Langflow package to run flow from JSON file
* Fix type annotations and import errors
* Refactor tests and fix test data
---------
Co-authored-by: Rodrigo Nader <rodrigosilvanader@gmail.com>
Co-authored-by: anovazzi1 <otavio2204@gmail.com>
Co-authored-by: Lucas Oliveira <lucas.edu.oli@hotmail.com>
Co-authored-by: carlosrcoelho <carlosrodrigo.coelho@gmail.com>
Co-authored-by: cristhianzl <cristhian.lousa@gmail.com>
Co-authored-by: Matheus <jacquesmats@gmail.com>
310 lines
11 KiB
Python
310 lines
11 KiB
Python
import warnings
|
|
from pathlib import Path
|
|
from typing import TYPE_CHECKING, Optional
|
|
|
|
from fastapi import HTTPException
|
|
from platformdirs import user_cache_dir
|
|
from sqlmodel import Session
|
|
|
|
from langflow.graph.graph.base import Graph
|
|
from langflow.services.chat.service import ChatService
|
|
from langflow.services.database.models.flow import Flow
|
|
from langflow.services.store.schema import StoreComponentCreate
|
|
from langflow.services.store.utils import get_lf_version_from_pypi
|
|
|
|
if TYPE_CHECKING:
|
|
from langflow.graph.vertex.base import Vertex
|
|
from langflow.services.database.models.flow.model import Flow
|
|
|
|
|
|
API_WORDS = ["api", "key", "token"]
|
|
|
|
|
|
def has_api_terms(word: str):
|
|
return "api" in word and ("key" in word or ("token" in word and "tokens" not in word))
|
|
|
|
|
|
def remove_api_keys(flow: dict):
|
|
"""Remove api keys from flow data."""
|
|
if flow.get("data") and flow["data"].get("nodes"):
|
|
for node in flow["data"]["nodes"]:
|
|
node_data = node.get("data").get("node")
|
|
template = node_data.get("template")
|
|
for value in template.values():
|
|
if isinstance(value, dict) and has_api_terms(value["name"]) and value.get("password"):
|
|
value["value"] = None
|
|
|
|
return flow
|
|
|
|
|
|
def build_input_keys_response(langchain_object, artifacts):
|
|
"""Build the input keys response."""
|
|
|
|
input_keys_response = {
|
|
"input_keys": {key: "" for key in langchain_object.input_keys},
|
|
"memory_keys": [],
|
|
"handle_keys": artifacts.get("handle_keys", []),
|
|
}
|
|
|
|
# Set the input keys values from artifacts
|
|
for key, value in artifacts.items():
|
|
if key in input_keys_response["input_keys"]:
|
|
input_keys_response["input_keys"][key] = value
|
|
# If the object has memory, that memory will have a memory_variables attribute
|
|
# memory variables should be removed from the input keys
|
|
if hasattr(langchain_object, "memory") and hasattr(langchain_object.memory, "memory_variables"):
|
|
# Remove memory variables from input keys
|
|
input_keys_response["input_keys"] = {
|
|
key: value
|
|
for key, value in input_keys_response["input_keys"].items()
|
|
if key not in langchain_object.memory.memory_variables
|
|
}
|
|
# Add memory variables to memory_keys
|
|
input_keys_response["memory_keys"] = langchain_object.memory.memory_variables
|
|
|
|
if hasattr(langchain_object, "prompt") and hasattr(langchain_object.prompt, "template"):
|
|
input_keys_response["template"] = langchain_object.prompt.template
|
|
|
|
return input_keys_response
|
|
|
|
|
|
def update_frontend_node_with_template_values(frontend_node, raw_frontend_node):
|
|
"""
|
|
Updates the given frontend node with values from the raw template data.
|
|
|
|
:param frontend_node: A dict representing a built frontend node.
|
|
:param raw_template_data: A dict representing raw template data.
|
|
:return: Updated frontend node.
|
|
"""
|
|
if not is_valid_data(frontend_node, raw_frontend_node):
|
|
return frontend_node
|
|
|
|
# Check if the display_name is different than "CustomComponent"
|
|
# if so, update the display_name in the frontend_node
|
|
if raw_frontend_node["display_name"] != "CustomComponent":
|
|
frontend_node["display_name"] = raw_frontend_node["display_name"]
|
|
|
|
update_template_values(frontend_node["template"], raw_frontend_node["template"])
|
|
|
|
return frontend_node
|
|
|
|
|
|
def raw_frontend_data_is_valid(raw_frontend_data):
|
|
"""Check if the raw frontend data is valid for processing."""
|
|
return "template" in raw_frontend_data and "display_name" in raw_frontend_data
|
|
|
|
|
|
def is_valid_data(frontend_node, raw_frontend_data):
|
|
"""Check if the data is valid for processing."""
|
|
|
|
return frontend_node and "template" in frontend_node and raw_frontend_data_is_valid(raw_frontend_data)
|
|
|
|
|
|
def update_template_values(frontend_template, raw_template):
|
|
"""Updates the frontend template with values from the raw template."""
|
|
for key, value_dict in raw_template.items():
|
|
if key == "code" or not isinstance(value_dict, dict):
|
|
continue
|
|
|
|
update_template_field(frontend_template, key, value_dict)
|
|
|
|
|
|
def update_template_field(frontend_template, key, value_dict):
|
|
"""Updates a specific field in the frontend template."""
|
|
template_field = frontend_template.get(key)
|
|
if not template_field or template_field.get("type") != value_dict.get("type"):
|
|
return
|
|
|
|
if "value" in value_dict and value_dict["value"]:
|
|
template_field["value"] = value_dict["value"]
|
|
|
|
if "file_path" in value_dict and value_dict["file_path"]:
|
|
file_path_value = get_file_path_value(value_dict["file_path"])
|
|
if not file_path_value:
|
|
# If the file does not exist, remove the value from the template_field["value"]
|
|
template_field["value"] = ""
|
|
template_field["file_path"] = file_path_value
|
|
|
|
if "load_from_db" in value_dict and value_dict["load_from_db"]:
|
|
template_field["load_from_db"] = value_dict["load_from_db"]
|
|
|
|
|
|
def get_file_path_value(file_path):
|
|
"""Get the file path value if the file exists, else return empty string."""
|
|
try:
|
|
path = Path(file_path)
|
|
except TypeError:
|
|
return ""
|
|
|
|
# Check for safety
|
|
# If the path is not in the cache dir, return empty string
|
|
# This is to prevent access to files outside the cache dir
|
|
# If the path is not a file, return empty string
|
|
if not path.exists() or not str(path).startswith(user_cache_dir("langflow", "langflow")):
|
|
return ""
|
|
return file_path
|
|
|
|
|
|
def validate_is_component(flows: list["Flow"]):
|
|
for flow in flows:
|
|
if not flow.data or flow.is_component is not None:
|
|
continue
|
|
|
|
is_component = get_is_component_from_data(flow.data)
|
|
if is_component is not None:
|
|
flow.is_component = is_component
|
|
else:
|
|
flow.is_component = len(flow.data.get("nodes", [])) == 1
|
|
return flows
|
|
|
|
|
|
def get_is_component_from_data(data: dict):
|
|
"""Returns True if the data is a component."""
|
|
return data.get("is_component")
|
|
|
|
|
|
async def check_langflow_version(component: StoreComponentCreate):
|
|
from langflow.version.version import __version__ as current_version # type: ignore
|
|
|
|
if not component.last_tested_version:
|
|
component.last_tested_version = current_version
|
|
|
|
langflow_version = get_lf_version_from_pypi()
|
|
if langflow_version is None:
|
|
raise HTTPException(status_code=500, detail="Unable to verify the latest version of Langflow")
|
|
elif langflow_version != component.last_tested_version:
|
|
warnings.warn(
|
|
f"Your version of Langflow ({component.last_tested_version}) is outdated. "
|
|
f"Please update to the latest version ({langflow_version}) and try again."
|
|
)
|
|
|
|
|
|
def format_elapsed_time(elapsed_time: float) -> str:
|
|
"""Format elapsed time to a human-readable format coming from perf_counter().
|
|
|
|
- Less than 1 second: returns milliseconds
|
|
- Less than 1 minute: returns seconds rounded to 2 decimals
|
|
- 1 minute or more: returns minutes and seconds
|
|
"""
|
|
if elapsed_time < 1:
|
|
milliseconds = int(round(elapsed_time * 1000))
|
|
return f"{milliseconds} ms"
|
|
elif elapsed_time < 60:
|
|
seconds = round(elapsed_time, 2)
|
|
unit = "second" if seconds == 1 else "seconds"
|
|
return f"{seconds} {unit}"
|
|
else:
|
|
minutes = int(elapsed_time // 60)
|
|
seconds = round(elapsed_time % 60, 2)
|
|
minutes_unit = "minute" if minutes == 1 else "minutes"
|
|
seconds_unit = "second" if seconds == 1 else "seconds"
|
|
return f"{minutes} {minutes_unit}, {seconds} {seconds_unit}"
|
|
|
|
|
|
async def build_and_cache_graph(
|
|
flow_id: str,
|
|
session: Session,
|
|
chat_service: "ChatService",
|
|
graph: Optional[Graph] = None,
|
|
):
|
|
"""Build and cache the graph."""
|
|
flow: Optional[Flow] = session.get(Flow, flow_id)
|
|
if not flow or not flow.data:
|
|
raise ValueError("Invalid flow ID")
|
|
other_graph = Graph.from_payload(flow.data, flow_id)
|
|
if graph is None:
|
|
graph = other_graph
|
|
else:
|
|
graph = graph.update(other_graph)
|
|
await chat_service.set_cache(flow_id, graph)
|
|
return graph
|
|
|
|
|
|
def format_syntax_error_message(exc: SyntaxError) -> str:
|
|
"""Format a SyntaxError message for returning to the frontend."""
|
|
if exc.text is None:
|
|
return f"Syntax error in code. Error on line {exc.lineno}"
|
|
return f"Syntax error in code. Error on line {exc.lineno}: {exc.text.strip()}"
|
|
|
|
|
|
def get_causing_exception(exc: BaseException) -> BaseException:
|
|
"""Get the causing exception from an exception."""
|
|
if hasattr(exc, "__cause__") and exc.__cause__:
|
|
return get_causing_exception(exc.__cause__)
|
|
return exc
|
|
|
|
|
|
def format_exception_message(exc: Exception) -> str:
|
|
"""Format an exception message for returning to the frontend."""
|
|
# We need to check if the __cause__ is a SyntaxError
|
|
# If it is, we need to return the message of the SyntaxError
|
|
causing_exception = get_causing_exception(exc)
|
|
if isinstance(causing_exception, SyntaxError):
|
|
return format_syntax_error_message(causing_exception)
|
|
return str(exc)
|
|
|
|
|
|
async def get_next_runnable_vertices(
|
|
graph: Graph,
|
|
vertex: "Vertex",
|
|
vertex_id: str,
|
|
chat_service: ChatService,
|
|
flow_id: str,
|
|
):
|
|
"""
|
|
Retrieves the next runnable vertices in the graph for a given vertex.
|
|
|
|
Args:
|
|
graph (Graph): The graph object representing the flow.
|
|
vertex (Vertex): The current vertex.
|
|
vertex_id (str): The ID of the current vertex.
|
|
chat_service (ChatService): The chat service object.
|
|
flow_id (str): The ID of the flow.
|
|
|
|
Returns:
|
|
list: A list of IDs of the next runnable vertices.
|
|
|
|
"""
|
|
async with chat_service._cache_locks[flow_id] as lock:
|
|
graph.remove_from_predecessors(vertex_id)
|
|
direct_successors_ready = [v for v in vertex.successors_ids if graph.is_vertex_runnable(v)]
|
|
if not direct_successors_ready:
|
|
# No direct successors ready, look for runnable predecessors of successors
|
|
next_runnable_vertices = graph.find_runnable_predecessors_for_successors(vertex_id)
|
|
else:
|
|
next_runnable_vertices = direct_successors_ready
|
|
|
|
for v_id in set(next_runnable_vertices): # Use set to avoid duplicates
|
|
graph.vertices_to_run.remove(v_id)
|
|
graph.remove_from_predecessors(v_id)
|
|
await chat_service.set_cache(flow_id=flow_id, data=graph, lock=lock)
|
|
return next_runnable_vertices
|
|
|
|
|
|
def get_top_level_vertices(graph, vertices_ids):
|
|
"""
|
|
Retrieves the top-level vertices from the given graph based on the provided vertex IDs.
|
|
|
|
Args:
|
|
graph (Graph): The graph object containing the vertices.
|
|
vertices_ids (list): A list of vertex IDs.
|
|
|
|
Returns:
|
|
list: A list of top-level vertex IDs.
|
|
|
|
"""
|
|
top_level_vertices = []
|
|
for vertex_id in vertices_ids:
|
|
vertex = graph.get_vertex(vertex_id)
|
|
if vertex.parent_is_top_level:
|
|
top_level_vertices.append(vertex.parent_node_id)
|
|
else:
|
|
top_level_vertices.append(vertex_id)
|
|
return top_level_vertices
|
|
|
|
|
|
def parse_exception(exc):
|
|
"""Parse the exception message."""
|
|
if hasattr(exc, "body"):
|
|
return exc.body["message"]
|
|
return str(exc)
|