lfx-nightly 0.2.0.dev25__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of lfx-nightly might be problematic. Click here for more details.
- lfx/__init__.py +0 -0
- lfx/__main__.py +25 -0
- lfx/_assets/component_index.json +1 -0
- lfx/base/__init__.py +0 -0
- lfx/base/agents/__init__.py +0 -0
- lfx/base/agents/agent.py +375 -0
- lfx/base/agents/altk_base_agent.py +380 -0
- lfx/base/agents/altk_tool_wrappers.py +565 -0
- lfx/base/agents/callback.py +130 -0
- lfx/base/agents/context.py +109 -0
- lfx/base/agents/crewai/__init__.py +0 -0
- lfx/base/agents/crewai/crew.py +231 -0
- lfx/base/agents/crewai/tasks.py +12 -0
- lfx/base/agents/default_prompts.py +23 -0
- lfx/base/agents/errors.py +15 -0
- lfx/base/agents/events.py +430 -0
- lfx/base/agents/utils.py +237 -0
- lfx/base/astra_assistants/__init__.py +0 -0
- lfx/base/astra_assistants/util.py +171 -0
- lfx/base/chains/__init__.py +0 -0
- lfx/base/chains/model.py +19 -0
- lfx/base/composio/__init__.py +0 -0
- lfx/base/composio/composio_base.py +2584 -0
- lfx/base/compressors/__init__.py +0 -0
- lfx/base/compressors/model.py +60 -0
- lfx/base/constants.py +46 -0
- lfx/base/curl/__init__.py +0 -0
- lfx/base/curl/parse.py +188 -0
- lfx/base/data/__init__.py +5 -0
- lfx/base/data/base_file.py +810 -0
- lfx/base/data/docling_utils.py +338 -0
- lfx/base/data/storage_utils.py +192 -0
- lfx/base/data/utils.py +362 -0
- lfx/base/datastax/__init__.py +5 -0
- lfx/base/datastax/astradb_base.py +896 -0
- lfx/base/document_transformers/__init__.py +0 -0
- lfx/base/document_transformers/model.py +43 -0
- lfx/base/embeddings/__init__.py +0 -0
- lfx/base/embeddings/aiml_embeddings.py +62 -0
- lfx/base/embeddings/embeddings_class.py +113 -0
- lfx/base/embeddings/model.py +26 -0
- lfx/base/flow_processing/__init__.py +0 -0
- lfx/base/flow_processing/utils.py +86 -0
- lfx/base/huggingface/__init__.py +0 -0
- lfx/base/huggingface/model_bridge.py +133 -0
- lfx/base/io/__init__.py +0 -0
- lfx/base/io/chat.py +21 -0
- lfx/base/io/text.py +22 -0
- lfx/base/knowledge_bases/__init__.py +3 -0
- lfx/base/knowledge_bases/knowledge_base_utils.py +137 -0
- lfx/base/langchain_utilities/__init__.py +0 -0
- lfx/base/langchain_utilities/model.py +35 -0
- lfx/base/langchain_utilities/spider_constants.py +1 -0
- lfx/base/langwatch/__init__.py +0 -0
- lfx/base/langwatch/utils.py +18 -0
- lfx/base/mcp/__init__.py +0 -0
- lfx/base/mcp/constants.py +2 -0
- lfx/base/mcp/util.py +1659 -0
- lfx/base/memory/__init__.py +0 -0
- lfx/base/memory/memory.py +49 -0
- lfx/base/memory/model.py +38 -0
- lfx/base/models/__init__.py +3 -0
- lfx/base/models/aiml_constants.py +51 -0
- lfx/base/models/anthropic_constants.py +51 -0
- lfx/base/models/aws_constants.py +151 -0
- lfx/base/models/chat_result.py +76 -0
- lfx/base/models/cometapi_constants.py +54 -0
- lfx/base/models/google_generative_ai_constants.py +70 -0
- lfx/base/models/google_generative_ai_model.py +38 -0
- lfx/base/models/groq_constants.py +150 -0
- lfx/base/models/groq_model_discovery.py +265 -0
- lfx/base/models/model.py +375 -0
- lfx/base/models/model_input_constants.py +378 -0
- lfx/base/models/model_metadata.py +41 -0
- lfx/base/models/model_utils.py +108 -0
- lfx/base/models/novita_constants.py +35 -0
- lfx/base/models/ollama_constants.py +52 -0
- lfx/base/models/openai_constants.py +129 -0
- lfx/base/models/sambanova_constants.py +18 -0
- lfx/base/models/watsonx_constants.py +36 -0
- lfx/base/processing/__init__.py +0 -0
- lfx/base/prompts/__init__.py +0 -0
- lfx/base/prompts/api_utils.py +224 -0
- lfx/base/prompts/utils.py +61 -0
- lfx/base/textsplitters/__init__.py +0 -0
- lfx/base/textsplitters/model.py +28 -0
- lfx/base/tools/__init__.py +0 -0
- lfx/base/tools/base.py +26 -0
- lfx/base/tools/component_tool.py +325 -0
- lfx/base/tools/constants.py +49 -0
- lfx/base/tools/flow_tool.py +132 -0
- lfx/base/tools/run_flow.py +698 -0
- lfx/base/vectorstores/__init__.py +0 -0
- lfx/base/vectorstores/model.py +193 -0
- lfx/base/vectorstores/utils.py +22 -0
- lfx/base/vectorstores/vector_store_connection_decorator.py +52 -0
- lfx/cli/__init__.py +5 -0
- lfx/cli/commands.py +327 -0
- lfx/cli/common.py +650 -0
- lfx/cli/run.py +506 -0
- lfx/cli/script_loader.py +289 -0
- lfx/cli/serve_app.py +546 -0
- lfx/cli/validation.py +69 -0
- lfx/components/FAISS/__init__.py +34 -0
- lfx/components/FAISS/faiss.py +111 -0
- lfx/components/Notion/__init__.py +19 -0
- lfx/components/Notion/add_content_to_page.py +269 -0
- lfx/components/Notion/create_page.py +94 -0
- lfx/components/Notion/list_database_properties.py +68 -0
- lfx/components/Notion/list_pages.py +122 -0
- lfx/components/Notion/list_users.py +77 -0
- lfx/components/Notion/page_content_viewer.py +93 -0
- lfx/components/Notion/search.py +111 -0
- lfx/components/Notion/update_page_property.py +114 -0
- lfx/components/__init__.py +428 -0
- lfx/components/_importing.py +42 -0
- lfx/components/agentql/__init__.py +3 -0
- lfx/components/agentql/agentql_api.py +151 -0
- lfx/components/aiml/__init__.py +37 -0
- lfx/components/aiml/aiml.py +115 -0
- lfx/components/aiml/aiml_embeddings.py +37 -0
- lfx/components/altk/__init__.py +34 -0
- lfx/components/altk/altk_agent.py +193 -0
- lfx/components/amazon/__init__.py +36 -0
- lfx/components/amazon/amazon_bedrock_converse.py +195 -0
- lfx/components/amazon/amazon_bedrock_embedding.py +109 -0
- lfx/components/amazon/amazon_bedrock_model.py +130 -0
- lfx/components/amazon/s3_bucket_uploader.py +211 -0
- lfx/components/anthropic/__init__.py +34 -0
- lfx/components/anthropic/anthropic.py +187 -0
- lfx/components/apify/__init__.py +5 -0
- lfx/components/apify/apify_actor.py +325 -0
- lfx/components/arxiv/__init__.py +3 -0
- lfx/components/arxiv/arxiv.py +169 -0
- lfx/components/assemblyai/__init__.py +46 -0
- lfx/components/assemblyai/assemblyai_get_subtitles.py +83 -0
- lfx/components/assemblyai/assemblyai_lemur.py +183 -0
- lfx/components/assemblyai/assemblyai_list_transcripts.py +95 -0
- lfx/components/assemblyai/assemblyai_poll_transcript.py +72 -0
- lfx/components/assemblyai/assemblyai_start_transcript.py +188 -0
- lfx/components/azure/__init__.py +37 -0
- lfx/components/azure/azure_openai.py +95 -0
- lfx/components/azure/azure_openai_embeddings.py +83 -0
- lfx/components/baidu/__init__.py +32 -0
- lfx/components/baidu/baidu_qianfan_chat.py +113 -0
- lfx/components/bing/__init__.py +3 -0
- lfx/components/bing/bing_search_api.py +61 -0
- lfx/components/cassandra/__init__.py +40 -0
- lfx/components/cassandra/cassandra.py +264 -0
- lfx/components/cassandra/cassandra_chat.py +92 -0
- lfx/components/cassandra/cassandra_graph.py +238 -0
- lfx/components/chains/__init__.py +3 -0
- lfx/components/chroma/__init__.py +34 -0
- lfx/components/chroma/chroma.py +169 -0
- lfx/components/cleanlab/__init__.py +40 -0
- lfx/components/cleanlab/cleanlab_evaluator.py +155 -0
- lfx/components/cleanlab/cleanlab_rag_evaluator.py +254 -0
- lfx/components/cleanlab/cleanlab_remediator.py +131 -0
- lfx/components/clickhouse/__init__.py +34 -0
- lfx/components/clickhouse/clickhouse.py +135 -0
- lfx/components/cloudflare/__init__.py +32 -0
- lfx/components/cloudflare/cloudflare.py +81 -0
- lfx/components/cohere/__init__.py +40 -0
- lfx/components/cohere/cohere_embeddings.py +81 -0
- lfx/components/cohere/cohere_models.py +46 -0
- lfx/components/cohere/cohere_rerank.py +51 -0
- lfx/components/cometapi/__init__.py +32 -0
- lfx/components/cometapi/cometapi.py +166 -0
- lfx/components/composio/__init__.py +222 -0
- lfx/components/composio/agentql_composio.py +11 -0
- lfx/components/composio/agiled_composio.py +11 -0
- lfx/components/composio/airtable_composio.py +11 -0
- lfx/components/composio/apollo_composio.py +11 -0
- lfx/components/composio/asana_composio.py +11 -0
- lfx/components/composio/attio_composio.py +11 -0
- lfx/components/composio/bitbucket_composio.py +11 -0
- lfx/components/composio/bolna_composio.py +11 -0
- lfx/components/composio/brightdata_composio.py +11 -0
- lfx/components/composio/calendly_composio.py +11 -0
- lfx/components/composio/canva_composio.py +11 -0
- lfx/components/composio/canvas_composio.py +11 -0
- lfx/components/composio/coda_composio.py +11 -0
- lfx/components/composio/composio_api.py +278 -0
- lfx/components/composio/contentful_composio.py +11 -0
- lfx/components/composio/digicert_composio.py +11 -0
- lfx/components/composio/discord_composio.py +11 -0
- lfx/components/composio/dropbox_compnent.py +11 -0
- lfx/components/composio/elevenlabs_composio.py +11 -0
- lfx/components/composio/exa_composio.py +11 -0
- lfx/components/composio/figma_composio.py +11 -0
- lfx/components/composio/finage_composio.py +11 -0
- lfx/components/composio/firecrawl_composio.py +11 -0
- lfx/components/composio/fireflies_composio.py +11 -0
- lfx/components/composio/fixer_composio.py +11 -0
- lfx/components/composio/flexisign_composio.py +11 -0
- lfx/components/composio/freshdesk_composio.py +11 -0
- lfx/components/composio/github_composio.py +11 -0
- lfx/components/composio/gmail_composio.py +38 -0
- lfx/components/composio/googlebigquery_composio.py +11 -0
- lfx/components/composio/googlecalendar_composio.py +11 -0
- lfx/components/composio/googleclassroom_composio.py +11 -0
- lfx/components/composio/googledocs_composio.py +11 -0
- lfx/components/composio/googlemeet_composio.py +11 -0
- lfx/components/composio/googlesheets_composio.py +11 -0
- lfx/components/composio/googletasks_composio.py +8 -0
- lfx/components/composio/heygen_composio.py +11 -0
- lfx/components/composio/instagram_composio.py +11 -0
- lfx/components/composio/jira_composio.py +11 -0
- lfx/components/composio/jotform_composio.py +11 -0
- lfx/components/composio/klaviyo_composio.py +11 -0
- lfx/components/composio/linear_composio.py +11 -0
- lfx/components/composio/listennotes_composio.py +11 -0
- lfx/components/composio/mem0_composio.py +11 -0
- lfx/components/composio/miro_composio.py +11 -0
- lfx/components/composio/missive_composio.py +11 -0
- lfx/components/composio/notion_composio.py +11 -0
- lfx/components/composio/onedrive_composio.py +11 -0
- lfx/components/composio/outlook_composio.py +11 -0
- lfx/components/composio/pandadoc_composio.py +11 -0
- lfx/components/composio/peopledatalabs_composio.py +11 -0
- lfx/components/composio/perplexityai_composio.py +11 -0
- lfx/components/composio/reddit_composio.py +11 -0
- lfx/components/composio/serpapi_composio.py +11 -0
- lfx/components/composio/slack_composio.py +11 -0
- lfx/components/composio/slackbot_composio.py +11 -0
- lfx/components/composio/snowflake_composio.py +11 -0
- lfx/components/composio/supabase_composio.py +11 -0
- lfx/components/composio/tavily_composio.py +11 -0
- lfx/components/composio/timelinesai_composio.py +11 -0
- lfx/components/composio/todoist_composio.py +11 -0
- lfx/components/composio/wrike_composio.py +11 -0
- lfx/components/composio/youtube_composio.py +11 -0
- lfx/components/confluence/__init__.py +3 -0
- lfx/components/confluence/confluence.py +84 -0
- lfx/components/couchbase/__init__.py +34 -0
- lfx/components/couchbase/couchbase.py +102 -0
- lfx/components/crewai/__init__.py +49 -0
- lfx/components/crewai/crewai.py +108 -0
- lfx/components/crewai/hierarchical_crew.py +47 -0
- lfx/components/crewai/hierarchical_task.py +45 -0
- lfx/components/crewai/sequential_crew.py +53 -0
- lfx/components/crewai/sequential_task.py +74 -0
- lfx/components/crewai/sequential_task_agent.py +144 -0
- lfx/components/cuga/__init__.py +34 -0
- lfx/components/cuga/cuga_agent.py +730 -0
- lfx/components/custom_component/__init__.py +34 -0
- lfx/components/custom_component/custom_component.py +31 -0
- lfx/components/data/__init__.py +114 -0
- lfx/components/data_source/__init__.py +58 -0
- lfx/components/data_source/api_request.py +577 -0
- lfx/components/data_source/csv_to_data.py +101 -0
- lfx/components/data_source/json_to_data.py +106 -0
- lfx/components/data_source/mock_data.py +398 -0
- lfx/components/data_source/news_search.py +166 -0
- lfx/components/data_source/rss.py +71 -0
- lfx/components/data_source/sql_executor.py +101 -0
- lfx/components/data_source/url.py +311 -0
- lfx/components/data_source/web_search.py +326 -0
- lfx/components/datastax/__init__.py +76 -0
- lfx/components/datastax/astradb_assistant_manager.py +307 -0
- lfx/components/datastax/astradb_chatmemory.py +40 -0
- lfx/components/datastax/astradb_cql.py +288 -0
- lfx/components/datastax/astradb_graph.py +217 -0
- lfx/components/datastax/astradb_tool.py +378 -0
- lfx/components/datastax/astradb_vectorize.py +122 -0
- lfx/components/datastax/astradb_vectorstore.py +449 -0
- lfx/components/datastax/create_assistant.py +59 -0
- lfx/components/datastax/create_thread.py +33 -0
- lfx/components/datastax/dotenv.py +36 -0
- lfx/components/datastax/get_assistant.py +38 -0
- lfx/components/datastax/getenvvar.py +31 -0
- lfx/components/datastax/graph_rag.py +141 -0
- lfx/components/datastax/hcd.py +315 -0
- lfx/components/datastax/list_assistants.py +26 -0
- lfx/components/datastax/run.py +90 -0
- lfx/components/deactivated/__init__.py +15 -0
- lfx/components/deactivated/amazon_kendra.py +66 -0
- lfx/components/deactivated/chat_litellm_model.py +158 -0
- lfx/components/deactivated/code_block_extractor.py +26 -0
- lfx/components/deactivated/documents_to_data.py +22 -0
- lfx/components/deactivated/embed.py +16 -0
- lfx/components/deactivated/extract_key_from_data.py +46 -0
- lfx/components/deactivated/json_document_builder.py +57 -0
- lfx/components/deactivated/list_flows.py +20 -0
- lfx/components/deactivated/mcp_sse.py +61 -0
- lfx/components/deactivated/mcp_stdio.py +62 -0
- lfx/components/deactivated/merge_data.py +93 -0
- lfx/components/deactivated/message.py +37 -0
- lfx/components/deactivated/metal.py +54 -0
- lfx/components/deactivated/multi_query.py +59 -0
- lfx/components/deactivated/retriever.py +43 -0
- lfx/components/deactivated/selective_passthrough.py +77 -0
- lfx/components/deactivated/should_run_next.py +40 -0
- lfx/components/deactivated/split_text.py +63 -0
- lfx/components/deactivated/store_message.py +24 -0
- lfx/components/deactivated/sub_flow.py +124 -0
- lfx/components/deactivated/vectara_self_query.py +76 -0
- lfx/components/deactivated/vector_store.py +24 -0
- lfx/components/deepseek/__init__.py +34 -0
- lfx/components/deepseek/deepseek.py +136 -0
- lfx/components/docling/__init__.py +43 -0
- lfx/components/docling/chunk_docling_document.py +186 -0
- lfx/components/docling/docling_inline.py +238 -0
- lfx/components/docling/docling_remote.py +195 -0
- lfx/components/docling/export_docling_document.py +117 -0
- lfx/components/documentloaders/__init__.py +3 -0
- lfx/components/duckduckgo/__init__.py +3 -0
- lfx/components/duckduckgo/duck_duck_go_search_run.py +92 -0
- lfx/components/elastic/__init__.py +37 -0
- lfx/components/elastic/elasticsearch.py +267 -0
- lfx/components/elastic/opensearch.py +789 -0
- lfx/components/elastic/opensearch_multimodal.py +1575 -0
- lfx/components/embeddings/__init__.py +37 -0
- lfx/components/embeddings/similarity.py +77 -0
- lfx/components/embeddings/text_embedder.py +65 -0
- lfx/components/exa/__init__.py +3 -0
- lfx/components/exa/exa_search.py +68 -0
- lfx/components/files_and_knowledge/__init__.py +47 -0
- lfx/components/files_and_knowledge/directory.py +113 -0
- lfx/components/files_and_knowledge/file.py +841 -0
- lfx/components/files_and_knowledge/ingestion.py +694 -0
- lfx/components/files_and_knowledge/retrieval.py +264 -0
- lfx/components/files_and_knowledge/save_file.py +746 -0
- lfx/components/firecrawl/__init__.py +43 -0
- lfx/components/firecrawl/firecrawl_crawl_api.py +88 -0
- lfx/components/firecrawl/firecrawl_extract_api.py +136 -0
- lfx/components/firecrawl/firecrawl_map_api.py +89 -0
- lfx/components/firecrawl/firecrawl_scrape_api.py +73 -0
- lfx/components/flow_controls/__init__.py +58 -0
- lfx/components/flow_controls/conditional_router.py +208 -0
- lfx/components/flow_controls/data_conditional_router.py +126 -0
- lfx/components/flow_controls/flow_tool.py +111 -0
- lfx/components/flow_controls/listen.py +29 -0
- lfx/components/flow_controls/loop.py +163 -0
- lfx/components/flow_controls/notify.py +88 -0
- lfx/components/flow_controls/pass_message.py +36 -0
- lfx/components/flow_controls/run_flow.py +108 -0
- lfx/components/flow_controls/sub_flow.py +115 -0
- lfx/components/git/__init__.py +4 -0
- lfx/components/git/git.py +262 -0
- lfx/components/git/gitextractor.py +196 -0
- lfx/components/glean/__init__.py +3 -0
- lfx/components/glean/glean_search_api.py +173 -0
- lfx/components/google/__init__.py +17 -0
- lfx/components/google/gmail.py +193 -0
- lfx/components/google/google_bq_sql_executor.py +157 -0
- lfx/components/google/google_drive.py +92 -0
- lfx/components/google/google_drive_search.py +152 -0
- lfx/components/google/google_generative_ai.py +144 -0
- lfx/components/google/google_generative_ai_embeddings.py +141 -0
- lfx/components/google/google_oauth_token.py +89 -0
- lfx/components/google/google_search_api_core.py +68 -0
- lfx/components/google/google_serper_api_core.py +74 -0
- lfx/components/groq/__init__.py +34 -0
- lfx/components/groq/groq.py +143 -0
- lfx/components/helpers/__init__.py +154 -0
- lfx/components/homeassistant/__init__.py +7 -0
- lfx/components/homeassistant/home_assistant_control.py +152 -0
- lfx/components/homeassistant/list_home_assistant_states.py +137 -0
- lfx/components/huggingface/__init__.py +37 -0
- lfx/components/huggingface/huggingface.py +199 -0
- lfx/components/huggingface/huggingface_inference_api.py +106 -0
- lfx/components/ibm/__init__.py +34 -0
- lfx/components/ibm/watsonx.py +207 -0
- lfx/components/ibm/watsonx_embeddings.py +135 -0
- lfx/components/icosacomputing/__init__.py +5 -0
- lfx/components/icosacomputing/combinatorial_reasoner.py +84 -0
- lfx/components/input_output/__init__.py +40 -0
- lfx/components/input_output/chat.py +109 -0
- lfx/components/input_output/chat_output.py +184 -0
- lfx/components/input_output/text.py +27 -0
- lfx/components/input_output/text_output.py +29 -0
- lfx/components/input_output/webhook.py +56 -0
- lfx/components/jigsawstack/__init__.py +23 -0
- lfx/components/jigsawstack/ai_scrape.py +126 -0
- lfx/components/jigsawstack/ai_web_search.py +136 -0
- lfx/components/jigsawstack/file_read.py +115 -0
- lfx/components/jigsawstack/file_upload.py +94 -0
- lfx/components/jigsawstack/image_generation.py +205 -0
- lfx/components/jigsawstack/nsfw.py +60 -0
- lfx/components/jigsawstack/object_detection.py +124 -0
- lfx/components/jigsawstack/sentiment.py +112 -0
- lfx/components/jigsawstack/text_to_sql.py +90 -0
- lfx/components/jigsawstack/text_translate.py +77 -0
- lfx/components/jigsawstack/vocr.py +107 -0
- lfx/components/knowledge_bases/__init__.py +89 -0
- lfx/components/langchain_utilities/__init__.py +109 -0
- lfx/components/langchain_utilities/character.py +53 -0
- lfx/components/langchain_utilities/conversation.py +59 -0
- lfx/components/langchain_utilities/csv_agent.py +175 -0
- lfx/components/langchain_utilities/fake_embeddings.py +26 -0
- lfx/components/langchain_utilities/html_link_extractor.py +35 -0
- lfx/components/langchain_utilities/json_agent.py +100 -0
- lfx/components/langchain_utilities/langchain_hub.py +126 -0
- lfx/components/langchain_utilities/language_recursive.py +49 -0
- lfx/components/langchain_utilities/language_semantic.py +138 -0
- lfx/components/langchain_utilities/llm_checker.py +39 -0
- lfx/components/langchain_utilities/llm_math.py +42 -0
- lfx/components/langchain_utilities/natural_language.py +61 -0
- lfx/components/langchain_utilities/openai_tools.py +53 -0
- lfx/components/langchain_utilities/openapi.py +48 -0
- lfx/components/langchain_utilities/recursive_character.py +60 -0
- lfx/components/langchain_utilities/retrieval_qa.py +83 -0
- lfx/components/langchain_utilities/runnable_executor.py +137 -0
- lfx/components/langchain_utilities/self_query.py +80 -0
- lfx/components/langchain_utilities/spider.py +142 -0
- lfx/components/langchain_utilities/sql.py +40 -0
- lfx/components/langchain_utilities/sql_database.py +35 -0
- lfx/components/langchain_utilities/sql_generator.py +78 -0
- lfx/components/langchain_utilities/tool_calling.py +59 -0
- lfx/components/langchain_utilities/vector_store_info.py +49 -0
- lfx/components/langchain_utilities/vector_store_router.py +33 -0
- lfx/components/langchain_utilities/xml_agent.py +71 -0
- lfx/components/langwatch/__init__.py +3 -0
- lfx/components/langwatch/langwatch.py +278 -0
- lfx/components/link_extractors/__init__.py +3 -0
- lfx/components/llm_operations/__init__.py +46 -0
- lfx/components/llm_operations/batch_run.py +205 -0
- lfx/components/llm_operations/lambda_filter.py +218 -0
- lfx/components/llm_operations/llm_conditional_router.py +421 -0
- lfx/components/llm_operations/llm_selector.py +499 -0
- lfx/components/llm_operations/structured_output.py +244 -0
- lfx/components/lmstudio/__init__.py +34 -0
- lfx/components/lmstudio/lmstudioembeddings.py +89 -0
- lfx/components/lmstudio/lmstudiomodel.py +133 -0
- lfx/components/logic/__init__.py +181 -0
- lfx/components/maritalk/__init__.py +32 -0
- lfx/components/maritalk/maritalk.py +52 -0
- lfx/components/mem0/__init__.py +3 -0
- lfx/components/mem0/mem0_chat_memory.py +147 -0
- lfx/components/milvus/__init__.py +34 -0
- lfx/components/milvus/milvus.py +115 -0
- lfx/components/mistral/__init__.py +37 -0
- lfx/components/mistral/mistral.py +114 -0
- lfx/components/mistral/mistral_embeddings.py +58 -0
- lfx/components/models/__init__.py +89 -0
- lfx/components/models_and_agents/__init__.py +49 -0
- lfx/components/models_and_agents/agent.py +644 -0
- lfx/components/models_and_agents/embedding_model.py +423 -0
- lfx/components/models_and_agents/language_model.py +398 -0
- lfx/components/models_and_agents/mcp_component.py +594 -0
- lfx/components/models_and_agents/memory.py +268 -0
- lfx/components/models_and_agents/prompt.py +67 -0
- lfx/components/mongodb/__init__.py +34 -0
- lfx/components/mongodb/mongodb_atlas.py +213 -0
- lfx/components/needle/__init__.py +3 -0
- lfx/components/needle/needle.py +104 -0
- lfx/components/notdiamond/__init__.py +34 -0
- lfx/components/notdiamond/notdiamond.py +228 -0
- lfx/components/novita/__init__.py +32 -0
- lfx/components/novita/novita.py +130 -0
- lfx/components/nvidia/__init__.py +57 -0
- lfx/components/nvidia/nvidia.py +151 -0
- lfx/components/nvidia/nvidia_embedding.py +77 -0
- lfx/components/nvidia/nvidia_ingest.py +317 -0
- lfx/components/nvidia/nvidia_rerank.py +63 -0
- lfx/components/nvidia/system_assist.py +65 -0
- lfx/components/olivya/__init__.py +3 -0
- lfx/components/olivya/olivya.py +116 -0
- lfx/components/ollama/__init__.py +37 -0
- lfx/components/ollama/ollama.py +548 -0
- lfx/components/ollama/ollama_embeddings.py +103 -0
- lfx/components/openai/__init__.py +37 -0
- lfx/components/openai/openai.py +100 -0
- lfx/components/openai/openai_chat_model.py +176 -0
- lfx/components/openrouter/__init__.py +32 -0
- lfx/components/openrouter/openrouter.py +104 -0
- lfx/components/output_parsers/__init__.py +3 -0
- lfx/components/perplexity/__init__.py +34 -0
- lfx/components/perplexity/perplexity.py +75 -0
- lfx/components/pgvector/__init__.py +34 -0
- lfx/components/pgvector/pgvector.py +72 -0
- lfx/components/pinecone/__init__.py +34 -0
- lfx/components/pinecone/pinecone.py +134 -0
- lfx/components/processing/__init__.py +72 -0
- lfx/components/processing/alter_metadata.py +109 -0
- lfx/components/processing/combine_text.py +40 -0
- lfx/components/processing/converter.py +248 -0
- lfx/components/processing/create_data.py +111 -0
- lfx/components/processing/create_list.py +40 -0
- lfx/components/processing/data_operations.py +528 -0
- lfx/components/processing/data_to_dataframe.py +71 -0
- lfx/components/processing/dataframe_operations.py +313 -0
- lfx/components/processing/dataframe_to_toolset.py +259 -0
- lfx/components/processing/dynamic_create_data.py +357 -0
- lfx/components/processing/extract_key.py +54 -0
- lfx/components/processing/filter_data.py +43 -0
- lfx/components/processing/filter_data_values.py +89 -0
- lfx/components/processing/json_cleaner.py +104 -0
- lfx/components/processing/merge_data.py +91 -0
- lfx/components/processing/message_to_data.py +37 -0
- lfx/components/processing/output_parser.py +46 -0
- lfx/components/processing/parse_data.py +71 -0
- lfx/components/processing/parse_dataframe.py +69 -0
- lfx/components/processing/parse_json_data.py +91 -0
- lfx/components/processing/parser.py +148 -0
- lfx/components/processing/regex.py +83 -0
- lfx/components/processing/select_data.py +49 -0
- lfx/components/processing/split_text.py +141 -0
- lfx/components/processing/store_message.py +91 -0
- lfx/components/processing/update_data.py +161 -0
- lfx/components/prototypes/__init__.py +35 -0
- lfx/components/prototypes/python_function.py +73 -0
- lfx/components/qdrant/__init__.py +34 -0
- lfx/components/qdrant/qdrant.py +109 -0
- lfx/components/redis/__init__.py +37 -0
- lfx/components/redis/redis.py +89 -0
- lfx/components/redis/redis_chat.py +43 -0
- lfx/components/sambanova/__init__.py +32 -0
- lfx/components/sambanova/sambanova.py +84 -0
- lfx/components/scrapegraph/__init__.py +40 -0
- lfx/components/scrapegraph/scrapegraph_markdownify_api.py +64 -0
- lfx/components/scrapegraph/scrapegraph_search_api.py +64 -0
- lfx/components/scrapegraph/scrapegraph_smart_scraper_api.py +71 -0
- lfx/components/searchapi/__init__.py +34 -0
- lfx/components/searchapi/search.py +79 -0
- lfx/components/serpapi/__init__.py +3 -0
- lfx/components/serpapi/serp.py +115 -0
- lfx/components/supabase/__init__.py +34 -0
- lfx/components/supabase/supabase.py +76 -0
- lfx/components/tavily/__init__.py +4 -0
- lfx/components/tavily/tavily_extract.py +117 -0
- lfx/components/tavily/tavily_search.py +212 -0
- lfx/components/textsplitters/__init__.py +3 -0
- lfx/components/toolkits/__init__.py +3 -0
- lfx/components/tools/__init__.py +66 -0
- lfx/components/tools/calculator.py +109 -0
- lfx/components/tools/google_search_api.py +45 -0
- lfx/components/tools/google_serper_api.py +115 -0
- lfx/components/tools/python_code_structured_tool.py +328 -0
- lfx/components/tools/python_repl.py +98 -0
- lfx/components/tools/search_api.py +88 -0
- lfx/components/tools/searxng.py +145 -0
- lfx/components/tools/serp_api.py +120 -0
- lfx/components/tools/tavily_search_tool.py +345 -0
- lfx/components/tools/wikidata_api.py +103 -0
- lfx/components/tools/wikipedia_api.py +50 -0
- lfx/components/tools/yahoo_finance.py +130 -0
- lfx/components/twelvelabs/__init__.py +52 -0
- lfx/components/twelvelabs/convert_astra_results.py +84 -0
- lfx/components/twelvelabs/pegasus_index.py +311 -0
- lfx/components/twelvelabs/split_video.py +301 -0
- lfx/components/twelvelabs/text_embeddings.py +57 -0
- lfx/components/twelvelabs/twelvelabs_pegasus.py +408 -0
- lfx/components/twelvelabs/video_embeddings.py +100 -0
- lfx/components/twelvelabs/video_file.py +191 -0
- lfx/components/unstructured/__init__.py +3 -0
- lfx/components/unstructured/unstructured.py +121 -0
- lfx/components/upstash/__init__.py +34 -0
- lfx/components/upstash/upstash.py +124 -0
- lfx/components/utilities/__init__.py +43 -0
- lfx/components/utilities/calculator_core.py +89 -0
- lfx/components/utilities/current_date.py +42 -0
- lfx/components/utilities/id_generator.py +42 -0
- lfx/components/utilities/python_repl_core.py +98 -0
- lfx/components/vectara/__init__.py +37 -0
- lfx/components/vectara/vectara.py +97 -0
- lfx/components/vectara/vectara_rag.py +164 -0
- lfx/components/vectorstores/__init__.py +34 -0
- lfx/components/vectorstores/local_db.py +270 -0
- lfx/components/vertexai/__init__.py +37 -0
- lfx/components/vertexai/vertexai.py +71 -0
- lfx/components/vertexai/vertexai_embeddings.py +67 -0
- lfx/components/vlmrun/__init__.py +34 -0
- lfx/components/vlmrun/vlmrun_transcription.py +224 -0
- lfx/components/weaviate/__init__.py +34 -0
- lfx/components/weaviate/weaviate.py +89 -0
- lfx/components/wikipedia/__init__.py +4 -0
- lfx/components/wikipedia/wikidata.py +86 -0
- lfx/components/wikipedia/wikipedia.py +53 -0
- lfx/components/wolframalpha/__init__.py +3 -0
- lfx/components/wolframalpha/wolfram_alpha_api.py +54 -0
- lfx/components/xai/__init__.py +32 -0
- lfx/components/xai/xai.py +167 -0
- lfx/components/yahoosearch/__init__.py +3 -0
- lfx/components/yahoosearch/yahoo.py +137 -0
- lfx/components/youtube/__init__.py +52 -0
- lfx/components/youtube/channel.py +227 -0
- lfx/components/youtube/comments.py +231 -0
- lfx/components/youtube/playlist.py +33 -0
- lfx/components/youtube/search.py +120 -0
- lfx/components/youtube/trending.py +285 -0
- lfx/components/youtube/video_details.py +263 -0
- lfx/components/youtube/youtube_transcripts.py +206 -0
- lfx/components/zep/__init__.py +3 -0
- lfx/components/zep/zep.py +45 -0
- lfx/constants.py +6 -0
- lfx/custom/__init__.py +7 -0
- lfx/custom/attributes.py +87 -0
- lfx/custom/code_parser/__init__.py +3 -0
- lfx/custom/code_parser/code_parser.py +361 -0
- lfx/custom/custom_component/__init__.py +0 -0
- lfx/custom/custom_component/base_component.py +128 -0
- lfx/custom/custom_component/component.py +1890 -0
- lfx/custom/custom_component/component_with_cache.py +8 -0
- lfx/custom/custom_component/custom_component.py +650 -0
- lfx/custom/dependency_analyzer.py +165 -0
- lfx/custom/directory_reader/__init__.py +3 -0
- lfx/custom/directory_reader/directory_reader.py +359 -0
- lfx/custom/directory_reader/utils.py +171 -0
- lfx/custom/eval.py +12 -0
- lfx/custom/schema.py +32 -0
- lfx/custom/tree_visitor.py +21 -0
- lfx/custom/utils.py +877 -0
- lfx/custom/validate.py +523 -0
- lfx/events/__init__.py +1 -0
- lfx/events/event_manager.py +110 -0
- lfx/exceptions/__init__.py +0 -0
- lfx/exceptions/component.py +15 -0
- lfx/field_typing/__init__.py +91 -0
- lfx/field_typing/constants.py +216 -0
- lfx/field_typing/range_spec.py +35 -0
- lfx/graph/__init__.py +6 -0
- lfx/graph/edge/__init__.py +0 -0
- lfx/graph/edge/base.py +300 -0
- lfx/graph/edge/schema.py +119 -0
- lfx/graph/edge/utils.py +0 -0
- lfx/graph/graph/__init__.py +0 -0
- lfx/graph/graph/ascii.py +202 -0
- lfx/graph/graph/base.py +2298 -0
- lfx/graph/graph/constants.py +63 -0
- lfx/graph/graph/runnable_vertices_manager.py +133 -0
- lfx/graph/graph/schema.py +53 -0
- lfx/graph/graph/state_model.py +66 -0
- lfx/graph/graph/utils.py +1024 -0
- lfx/graph/schema.py +75 -0
- lfx/graph/state/__init__.py +0 -0
- lfx/graph/state/model.py +250 -0
- lfx/graph/utils.py +206 -0
- lfx/graph/vertex/__init__.py +0 -0
- lfx/graph/vertex/base.py +826 -0
- lfx/graph/vertex/constants.py +0 -0
- lfx/graph/vertex/exceptions.py +4 -0
- lfx/graph/vertex/param_handler.py +316 -0
- lfx/graph/vertex/schema.py +26 -0
- lfx/graph/vertex/utils.py +19 -0
- lfx/graph/vertex/vertex_types.py +489 -0
- lfx/helpers/__init__.py +141 -0
- lfx/helpers/base_model.py +71 -0
- lfx/helpers/custom.py +13 -0
- lfx/helpers/data.py +167 -0
- lfx/helpers/flow.py +308 -0
- lfx/inputs/__init__.py +68 -0
- lfx/inputs/constants.py +2 -0
- lfx/inputs/input_mixin.py +352 -0
- lfx/inputs/inputs.py +718 -0
- lfx/inputs/validators.py +19 -0
- lfx/interface/__init__.py +6 -0
- lfx/interface/components.py +897 -0
- lfx/interface/importing/__init__.py +5 -0
- lfx/interface/importing/utils.py +39 -0
- lfx/interface/initialize/__init__.py +3 -0
- lfx/interface/initialize/loading.py +317 -0
- lfx/interface/listing.py +26 -0
- lfx/interface/run.py +16 -0
- lfx/interface/utils.py +111 -0
- lfx/io/__init__.py +63 -0
- lfx/io/schema.py +295 -0
- lfx/load/__init__.py +8 -0
- lfx/load/load.py +256 -0
- lfx/load/utils.py +99 -0
- lfx/log/__init__.py +5 -0
- lfx/log/logger.py +411 -0
- lfx/logging/__init__.py +11 -0
- lfx/logging/logger.py +24 -0
- lfx/memory/__init__.py +70 -0
- lfx/memory/stubs.py +302 -0
- lfx/processing/__init__.py +1 -0
- lfx/processing/process.py +238 -0
- lfx/processing/utils.py +25 -0
- lfx/py.typed +0 -0
- lfx/schema/__init__.py +66 -0
- lfx/schema/artifact.py +83 -0
- lfx/schema/content_block.py +62 -0
- lfx/schema/content_types.py +91 -0
- lfx/schema/cross_module.py +80 -0
- lfx/schema/data.py +309 -0
- lfx/schema/dataframe.py +210 -0
- lfx/schema/dotdict.py +74 -0
- lfx/schema/encoders.py +13 -0
- lfx/schema/graph.py +47 -0
- lfx/schema/image.py +184 -0
- lfx/schema/json_schema.py +186 -0
- lfx/schema/log.py +62 -0
- lfx/schema/message.py +493 -0
- lfx/schema/openai_responses_schemas.py +74 -0
- lfx/schema/properties.py +41 -0
- lfx/schema/schema.py +180 -0
- lfx/schema/serialize.py +13 -0
- lfx/schema/table.py +142 -0
- lfx/schema/validators.py +114 -0
- lfx/serialization/__init__.py +5 -0
- lfx/serialization/constants.py +2 -0
- lfx/serialization/serialization.py +314 -0
- lfx/services/__init__.py +26 -0
- lfx/services/base.py +28 -0
- lfx/services/cache/__init__.py +6 -0
- lfx/services/cache/base.py +183 -0
- lfx/services/cache/service.py +166 -0
- lfx/services/cache/utils.py +169 -0
- lfx/services/chat/__init__.py +1 -0
- lfx/services/chat/config.py +2 -0
- lfx/services/chat/schema.py +10 -0
- lfx/services/database/__init__.py +5 -0
- lfx/services/database/service.py +25 -0
- lfx/services/deps.py +194 -0
- lfx/services/factory.py +19 -0
- lfx/services/initialize.py +19 -0
- lfx/services/interfaces.py +103 -0
- lfx/services/manager.py +185 -0
- lfx/services/mcp_composer/__init__.py +6 -0
- lfx/services/mcp_composer/factory.py +16 -0
- lfx/services/mcp_composer/service.py +1441 -0
- lfx/services/schema.py +21 -0
- lfx/services/session.py +87 -0
- lfx/services/settings/__init__.py +3 -0
- lfx/services/settings/auth.py +133 -0
- lfx/services/settings/base.py +668 -0
- lfx/services/settings/constants.py +43 -0
- lfx/services/settings/factory.py +23 -0
- lfx/services/settings/feature_flags.py +11 -0
- lfx/services/settings/service.py +35 -0
- lfx/services/settings/utils.py +40 -0
- lfx/services/shared_component_cache/__init__.py +1 -0
- lfx/services/shared_component_cache/factory.py +30 -0
- lfx/services/shared_component_cache/service.py +9 -0
- lfx/services/storage/__init__.py +5 -0
- lfx/services/storage/local.py +185 -0
- lfx/services/storage/service.py +177 -0
- lfx/services/tracing/__init__.py +1 -0
- lfx/services/tracing/service.py +21 -0
- lfx/settings.py +6 -0
- lfx/template/__init__.py +6 -0
- lfx/template/field/__init__.py +0 -0
- lfx/template/field/base.py +260 -0
- lfx/template/field/prompt.py +15 -0
- lfx/template/frontend_node/__init__.py +6 -0
- lfx/template/frontend_node/base.py +214 -0
- lfx/template/frontend_node/constants.py +65 -0
- lfx/template/frontend_node/custom_components.py +79 -0
- lfx/template/template/__init__.py +0 -0
- lfx/template/template/base.py +100 -0
- lfx/template/utils.py +217 -0
- lfx/type_extraction/__init__.py +19 -0
- lfx/type_extraction/type_extraction.py +75 -0
- lfx/type_extraction.py +80 -0
- lfx/utils/__init__.py +1 -0
- lfx/utils/async_helpers.py +42 -0
- lfx/utils/component_utils.py +154 -0
- lfx/utils/concurrency.py +60 -0
- lfx/utils/connection_string_parser.py +11 -0
- lfx/utils/constants.py +233 -0
- lfx/utils/data_structure.py +212 -0
- lfx/utils/exceptions.py +22 -0
- lfx/utils/helpers.py +34 -0
- lfx/utils/image.py +79 -0
- lfx/utils/langflow_utils.py +52 -0
- lfx/utils/lazy_load.py +15 -0
- lfx/utils/request_utils.py +18 -0
- lfx/utils/schemas.py +139 -0
- lfx/utils/ssrf_protection.py +384 -0
- lfx/utils/util.py +626 -0
- lfx/utils/util_strings.py +56 -0
- lfx/utils/validate_cloud.py +26 -0
- lfx/utils/version.py +24 -0
- lfx_nightly-0.2.0.dev25.dist-info/METADATA +312 -0
- lfx_nightly-0.2.0.dev25.dist-info/RECORD +769 -0
- lfx_nightly-0.2.0.dev25.dist-info/WHEEL +4 -0
- lfx_nightly-0.2.0.dev25.dist-info/entry_points.txt +2 -0
|
@@ -0,0 +1,668 @@
|
|
|
1
|
+
import asyncio
|
|
2
|
+
import contextlib
|
|
3
|
+
import json
|
|
4
|
+
import os
|
|
5
|
+
from pathlib import Path
|
|
6
|
+
from shutil import copy2
|
|
7
|
+
from typing import Any, Literal
|
|
8
|
+
|
|
9
|
+
import orjson
|
|
10
|
+
import yaml
|
|
11
|
+
from aiofile import async_open
|
|
12
|
+
from pydantic import Field, field_validator
|
|
13
|
+
from pydantic.fields import FieldInfo
|
|
14
|
+
from pydantic_settings import BaseSettings, EnvSettingsSource, PydanticBaseSettingsSource, SettingsConfigDict
|
|
15
|
+
from typing_extensions import override
|
|
16
|
+
|
|
17
|
+
from lfx.constants import BASE_COMPONENTS_PATH
|
|
18
|
+
from lfx.log.logger import logger
|
|
19
|
+
from lfx.serialization.constants import MAX_ITEMS_LENGTH, MAX_TEXT_LENGTH
|
|
20
|
+
from lfx.services.settings.constants import AGENTIC_VARIABLES, VARIABLES_TO_GET_FROM_ENVIRONMENT
|
|
21
|
+
from lfx.utils.util_strings import is_valid_database_url
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
def is_list_of_any(field: FieldInfo) -> bool:
|
|
25
|
+
"""Check if the given field is a list or an optional list of any type.
|
|
26
|
+
|
|
27
|
+
Args:
|
|
28
|
+
field (FieldInfo): The field to be checked.
|
|
29
|
+
|
|
30
|
+
Returns:
|
|
31
|
+
bool: True if the field is a list or a list of any type, False otherwise.
|
|
32
|
+
"""
|
|
33
|
+
if field.annotation is None:
|
|
34
|
+
return False
|
|
35
|
+
try:
|
|
36
|
+
union_args = field.annotation.__args__ if hasattr(field.annotation, "__args__") else []
|
|
37
|
+
|
|
38
|
+
return field.annotation.__origin__ is list or any(
|
|
39
|
+
arg.__origin__ is list for arg in union_args if hasattr(arg, "__origin__")
|
|
40
|
+
)
|
|
41
|
+
except AttributeError:
|
|
42
|
+
return False
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
class CustomSource(EnvSettingsSource):
|
|
46
|
+
@override
|
|
47
|
+
def prepare_field_value(self, field_name: str, field: FieldInfo, value: Any, value_is_complex: bool) -> Any: # type: ignore[misc]
|
|
48
|
+
# allow comma-separated list parsing
|
|
49
|
+
|
|
50
|
+
# fieldInfo contains the annotation of the field
|
|
51
|
+
if is_list_of_any(field):
|
|
52
|
+
if isinstance(value, str):
|
|
53
|
+
value = value.split(",")
|
|
54
|
+
if isinstance(value, list):
|
|
55
|
+
return value
|
|
56
|
+
|
|
57
|
+
return super().prepare_field_value(field_name, field, value, value_is_complex)
|
|
58
|
+
|
|
59
|
+
|
|
60
|
+
class Settings(BaseSettings):
|
|
61
|
+
# Define the default LANGFLOW_DIR
|
|
62
|
+
config_dir: str | None = None
|
|
63
|
+
# Define if langflow db should be saved in config dir or
|
|
64
|
+
# in the langflow directory
|
|
65
|
+
save_db_in_config_dir: bool = False
|
|
66
|
+
"""Define if langflow database should be saved in LANGFLOW_CONFIG_DIR or in the langflow directory
|
|
67
|
+
(i.e. in the package directory)."""
|
|
68
|
+
|
|
69
|
+
knowledge_bases_dir: str | None = "~/.langflow/knowledge_bases"
|
|
70
|
+
"""The directory to store knowledge bases."""
|
|
71
|
+
|
|
72
|
+
dev: bool = False
|
|
73
|
+
"""If True, Langflow will run in development mode."""
|
|
74
|
+
database_url: str | None = None
|
|
75
|
+
"""Database URL for Langflow. If not provided, Langflow will use a SQLite database.
|
|
76
|
+
The driver shall be an async one like `sqlite+aiosqlite` (`sqlite` and `postgresql`
|
|
77
|
+
will be automatically converted to the async drivers `sqlite+aiosqlite` and
|
|
78
|
+
`postgresql+psycopg` respectively)."""
|
|
79
|
+
database_connection_retry: bool = False
|
|
80
|
+
"""If True, Langflow will retry to connect to the database if it fails."""
|
|
81
|
+
pool_size: int = 20
|
|
82
|
+
"""The number of connections to keep open in the connection pool.
|
|
83
|
+
For high load scenarios, this should be increased based on expected concurrent users."""
|
|
84
|
+
max_overflow: int = 30
|
|
85
|
+
"""The number of connections to allow that can be opened beyond the pool size.
|
|
86
|
+
Should be 2x the pool_size for optimal performance under load."""
|
|
87
|
+
db_connect_timeout: int = 30
|
|
88
|
+
"""The number of seconds to wait before giving up on a lock to released or establishing a connection to the
|
|
89
|
+
database."""
|
|
90
|
+
migration_lock_namespace: str | None = None
|
|
91
|
+
"""Optional namespace identifier for PostgreSQL advisory lock during migrations.
|
|
92
|
+
If not provided, a hash of the database URL will be used. Useful when multiple Langflow
|
|
93
|
+
instances share the same database and need coordinated migration locking."""
|
|
94
|
+
|
|
95
|
+
mcp_server_timeout: int = 20
|
|
96
|
+
"""The number of seconds to wait before giving up on a lock to released or establishing a connection to the
|
|
97
|
+
database."""
|
|
98
|
+
|
|
99
|
+
# ---------------------------------------------------------------------
|
|
100
|
+
# MCP Session-manager tuning
|
|
101
|
+
# ---------------------------------------------------------------------
|
|
102
|
+
mcp_max_sessions_per_server: int = 10
|
|
103
|
+
"""Maximum number of MCP sessions to keep per unique server (command/url).
|
|
104
|
+
Mirrors the default constant MAX_SESSIONS_PER_SERVER in util.py. Adjust to
|
|
105
|
+
control resource usage or concurrency per server."""
|
|
106
|
+
|
|
107
|
+
mcp_session_idle_timeout: int = 400 # seconds
|
|
108
|
+
"""How long (in seconds) an MCP session can stay idle before the background
|
|
109
|
+
cleanup task disposes of it. Defaults to 5 minutes."""
|
|
110
|
+
|
|
111
|
+
mcp_session_cleanup_interval: int = 120 # seconds
|
|
112
|
+
"""Frequency (in seconds) at which the background cleanup task wakes up to
|
|
113
|
+
reap idle sessions."""
|
|
114
|
+
|
|
115
|
+
# sqlite configuration
|
|
116
|
+
sqlite_pragmas: dict | None = {"synchronous": "NORMAL", "journal_mode": "WAL", "busy_timeout": 30000}
|
|
117
|
+
"""SQLite pragmas to use when connecting to the database."""
|
|
118
|
+
|
|
119
|
+
db_driver_connection_settings: dict | None = None
|
|
120
|
+
"""Database driver connection settings."""
|
|
121
|
+
|
|
122
|
+
db_connection_settings: dict | None = {
|
|
123
|
+
"pool_size": 20, # Match the pool_size above
|
|
124
|
+
"max_overflow": 30, # Match the max_overflow above
|
|
125
|
+
"pool_timeout": 30, # Seconds to wait for a connection from pool
|
|
126
|
+
"pool_pre_ping": True, # Check connection validity before using
|
|
127
|
+
"pool_recycle": 1800, # Recycle connections after 30 minutes
|
|
128
|
+
"echo": False, # Set to True for debugging only
|
|
129
|
+
}
|
|
130
|
+
"""Database connection settings optimized for high load scenarios.
|
|
131
|
+
Note: These settings are most effective with PostgreSQL. For SQLite:
|
|
132
|
+
- Reduce pool_size and max_overflow if experiencing lock contention
|
|
133
|
+
- SQLite has limited concurrent write capability even with WAL mode
|
|
134
|
+
- Best for read-heavy or moderate write workloads
|
|
135
|
+
|
|
136
|
+
Settings:
|
|
137
|
+
- pool_size: Number of connections to maintain (increase for higher concurrency)
|
|
138
|
+
- max_overflow: Additional connections allowed beyond pool_size
|
|
139
|
+
- pool_timeout: Seconds to wait for an available connection
|
|
140
|
+
- pool_pre_ping: Validates connections before use to prevent stale connections
|
|
141
|
+
- pool_recycle: Seconds before connections are recycled (prevents timeouts)
|
|
142
|
+
- echo: Enable SQL query logging (development only)
|
|
143
|
+
"""
|
|
144
|
+
|
|
145
|
+
use_noop_database: bool = False
|
|
146
|
+
"""If True, disables all database operations and uses a no-op session.
|
|
147
|
+
Controlled by LANGFLOW_USE_NOOP_DATABASE env variable."""
|
|
148
|
+
|
|
149
|
+
# cache configuration
|
|
150
|
+
cache_type: Literal["async", "redis", "memory", "disk"] = "async"
|
|
151
|
+
"""The cache type can be 'async' or 'redis'."""
|
|
152
|
+
cache_expire: int = 3600
|
|
153
|
+
"""The cache expire in seconds."""
|
|
154
|
+
variable_store: str = "db"
|
|
155
|
+
"""The store can be 'db' or 'kubernetes'."""
|
|
156
|
+
|
|
157
|
+
prometheus_enabled: bool = False
|
|
158
|
+
"""If set to True, Langflow will expose Prometheus metrics."""
|
|
159
|
+
prometheus_port: int = 9090
|
|
160
|
+
"""The port on which Langflow will expose Prometheus metrics. 9090 is the default port."""
|
|
161
|
+
|
|
162
|
+
disable_track_apikey_usage: bool = False
|
|
163
|
+
remove_api_keys: bool = False
|
|
164
|
+
components_path: list[str] = []
|
|
165
|
+
components_index_path: str | None = None
|
|
166
|
+
"""Path or URL to a prebuilt component index JSON file.
|
|
167
|
+
|
|
168
|
+
If None, uses the built-in index at lfx/_assets/component_index.json.
|
|
169
|
+
Set to a file path (e.g., '/path/to/index.json') or URL (e.g., 'https://example.com/index.json')
|
|
170
|
+
to use a custom index.
|
|
171
|
+
"""
|
|
172
|
+
langchain_cache: str = "InMemoryCache"
|
|
173
|
+
load_flows_path: str | None = None
|
|
174
|
+
bundle_urls: list[str] = []
|
|
175
|
+
|
|
176
|
+
# Redis
|
|
177
|
+
redis_host: str = "localhost"
|
|
178
|
+
redis_port: int = 6379
|
|
179
|
+
redis_db: int = 0
|
|
180
|
+
redis_url: str | None = None
|
|
181
|
+
redis_cache_expire: int = 3600
|
|
182
|
+
|
|
183
|
+
# Sentry
|
|
184
|
+
sentry_dsn: str | None = None
|
|
185
|
+
sentry_traces_sample_rate: float | None = 1.0
|
|
186
|
+
sentry_profiles_sample_rate: float | None = 1.0
|
|
187
|
+
|
|
188
|
+
store: bool | None = True
|
|
189
|
+
store_url: str | None = "https://api.langflow.store"
|
|
190
|
+
download_webhook_url: str | None = "https://api.langflow.store/flows/trigger/ec611a61-8460-4438-b187-a4f65e5559d4"
|
|
191
|
+
like_webhook_url: str | None = "https://api.langflow.store/flows/trigger/64275852-ec00-45c1-984e-3bff814732da"
|
|
192
|
+
|
|
193
|
+
storage_type: str = "local"
|
|
194
|
+
"""Storage type for file storage. Defaults to 'local'. Supports 'local' and 's3'."""
|
|
195
|
+
object_storage_bucket_name: str | None = "langflow-bucket"
|
|
196
|
+
"""Object storage bucket name for file storage. Defaults to 'langflow-bucket'."""
|
|
197
|
+
object_storage_prefix: str | None = "files"
|
|
198
|
+
"""Object storage prefix for file storage. Defaults to 'files'."""
|
|
199
|
+
object_storage_tags: dict[str, str] | None = None
|
|
200
|
+
"""Object storage tags for file storage."""
|
|
201
|
+
|
|
202
|
+
celery_enabled: bool = False
|
|
203
|
+
|
|
204
|
+
fallback_to_env_var: bool = True
|
|
205
|
+
"""If set to True, Global Variables set in the UI will fallback to a environment variable
|
|
206
|
+
with the same name in case Langflow fails to retrieve the variable value."""
|
|
207
|
+
|
|
208
|
+
store_environment_variables: bool = True
|
|
209
|
+
"""Whether to store environment variables as Global Variables in the database."""
|
|
210
|
+
variables_to_get_from_environment: list[str] = VARIABLES_TO_GET_FROM_ENVIRONMENT
|
|
211
|
+
"""List of environment variables to get from the environment and store in the database."""
|
|
212
|
+
worker_timeout: int = 300
|
|
213
|
+
"""Timeout for the API calls in seconds."""
|
|
214
|
+
frontend_timeout: int = 0
|
|
215
|
+
"""Timeout for the frontend API calls in seconds."""
|
|
216
|
+
user_agent: str = "langflow"
|
|
217
|
+
"""User agent for the API calls."""
|
|
218
|
+
backend_only: bool = False
|
|
219
|
+
"""If set to True, Langflow will not serve the frontend."""
|
|
220
|
+
|
|
221
|
+
# CORS Settings
|
|
222
|
+
cors_origins: list[str] | str = "*"
|
|
223
|
+
"""Allowed origins for CORS. Can be a list of origins or '*' for all origins.
|
|
224
|
+
Default is '*' for backward compatibility. In production, specify exact origins."""
|
|
225
|
+
cors_allow_credentials: bool = True
|
|
226
|
+
"""Whether to allow credentials in CORS requests.
|
|
227
|
+
Default is True for backward compatibility. In v2.0, this will be changed to False when using wildcard origins."""
|
|
228
|
+
cors_allow_methods: list[str] | str = "*"
|
|
229
|
+
"""Allowed HTTP methods for CORS requests."""
|
|
230
|
+
cors_allow_headers: list[str] | str = "*"
|
|
231
|
+
"""Allowed headers for CORS requests."""
|
|
232
|
+
|
|
233
|
+
# Telemetry
|
|
234
|
+
do_not_track: bool = False
|
|
235
|
+
"""If set to True, Langflow will not track telemetry."""
|
|
236
|
+
telemetry_base_url: str = "https://langflow.gateway.scarf.sh"
|
|
237
|
+
transactions_storage_enabled: bool = True
|
|
238
|
+
"""If set to True, Langflow will track transactions between flows."""
|
|
239
|
+
vertex_builds_storage_enabled: bool = True
|
|
240
|
+
"""If set to True, Langflow will keep track of each vertex builds (outputs) in the UI for any flow."""
|
|
241
|
+
|
|
242
|
+
# Config
|
|
243
|
+
host: str = "localhost"
|
|
244
|
+
"""The host on which Langflow will run."""
|
|
245
|
+
port: int = 7860
|
|
246
|
+
"""The port on which Langflow will run."""
|
|
247
|
+
runtime_port: int | None = Field(default=None, exclude=True)
|
|
248
|
+
"""TEMPORARY: The port detected at runtime after checking for conflicts.
|
|
249
|
+
This field is system-managed only and will be removed in future versions
|
|
250
|
+
when strict port enforcement is implemented (errors will be raised if port unavailable)."""
|
|
251
|
+
workers: int = 1
|
|
252
|
+
"""The number of workers to run."""
|
|
253
|
+
log_level: str = "critical"
|
|
254
|
+
"""The log level for Langflow."""
|
|
255
|
+
log_file: str | None = "logs/langflow.log"
|
|
256
|
+
"""The path to log file for Langflow."""
|
|
257
|
+
alembic_log_file: str = "alembic/alembic.log"
|
|
258
|
+
"""The path to log file for Alembic for SQLAlchemy."""
|
|
259
|
+
alembic_log_to_stdout: bool = False
|
|
260
|
+
"""If set to True, the log file will be ignored and Alembic will log to stdout."""
|
|
261
|
+
frontend_path: str | None = None
|
|
262
|
+
"""The path to the frontend directory containing build files. This is for development purposes only.."""
|
|
263
|
+
open_browser: bool = False
|
|
264
|
+
"""If set to True, Langflow will open the browser on startup."""
|
|
265
|
+
auto_saving: bool = True
|
|
266
|
+
"""If set to True, Langflow will auto save flows."""
|
|
267
|
+
auto_saving_interval: int = 1000
|
|
268
|
+
"""The interval in ms at which Langflow will auto save flows."""
|
|
269
|
+
health_check_max_retries: int = 5
|
|
270
|
+
"""The maximum number of retries for the health check."""
|
|
271
|
+
max_file_size_upload: int = 1024
|
|
272
|
+
"""The maximum file size for the upload in MB."""
|
|
273
|
+
deactivate_tracing: bool = False
|
|
274
|
+
"""If set to True, tracing will be deactivated."""
|
|
275
|
+
max_transactions_to_keep: int = 3000
|
|
276
|
+
"""The maximum number of transactions to keep in the database."""
|
|
277
|
+
max_vertex_builds_to_keep: int = 3000
|
|
278
|
+
"""The maximum number of vertex builds to keep in the database."""
|
|
279
|
+
max_vertex_builds_per_vertex: int = 2
|
|
280
|
+
"""The maximum number of builds to keep per vertex. Older builds will be deleted."""
|
|
281
|
+
webhook_polling_interval: int = 5000
|
|
282
|
+
"""The polling interval for the webhook in ms."""
|
|
283
|
+
fs_flows_polling_interval: int = 10000
|
|
284
|
+
"""The polling interval in milliseconds for synchronizing flows from the file system."""
|
|
285
|
+
ssl_cert_file: str | None = None
|
|
286
|
+
"""Path to the SSL certificate file on the local system."""
|
|
287
|
+
ssl_key_file: str | None = None
|
|
288
|
+
"""Path to the SSL key file on the local system."""
|
|
289
|
+
max_text_length: int = MAX_TEXT_LENGTH
|
|
290
|
+
"""Maximum number of characters to store and display in the UI. Responses longer than this
|
|
291
|
+
will be truncated when displayed in the UI. Does not truncate responses between components nor outputs."""
|
|
292
|
+
max_items_length: int = MAX_ITEMS_LENGTH
|
|
293
|
+
"""Maximum number of items to store and display in the UI. Lists longer than this
|
|
294
|
+
will be truncated when displayed in the UI. Does not affect data passed between components nor outputs."""
|
|
295
|
+
|
|
296
|
+
# MCP Server
|
|
297
|
+
mcp_server_enabled: bool = True
|
|
298
|
+
"""If set to False, Langflow will not enable the MCP server."""
|
|
299
|
+
mcp_server_enable_progress_notifications: bool = False
|
|
300
|
+
"""If set to False, Langflow will not send progress notifications in the MCP server."""
|
|
301
|
+
|
|
302
|
+
# Add projects to MCP servers automatically on creation
|
|
303
|
+
add_projects_to_mcp_servers: bool = True
|
|
304
|
+
"""If set to True, newly created projects will be added to the user's MCP servers config automatically."""
|
|
305
|
+
# MCP Composer
|
|
306
|
+
mcp_composer_enabled: bool = True
|
|
307
|
+
"""If set to False, Langflow will not start the MCP Composer service."""
|
|
308
|
+
mcp_composer_version: str = "==0.1.0.8.10"
|
|
309
|
+
"""Version constraint for mcp-composer when using uvx. Uses PEP 440 syntax."""
|
|
310
|
+
|
|
311
|
+
# Agentic Experience
|
|
312
|
+
agentic_experience: bool = False
|
|
313
|
+
"""If set to True, Langflow will start the agentic MCP server that provides tools for
|
|
314
|
+
flow/component operations, template search, and graph visualization."""
|
|
315
|
+
|
|
316
|
+
# Public Flow Settings
|
|
317
|
+
public_flow_cleanup_interval: int = Field(default=3600, gt=600)
|
|
318
|
+
"""The interval in seconds at which public temporary flows will be cleaned up.
|
|
319
|
+
Default is 1 hour (3600 seconds). Minimum is 600 seconds (10 minutes)."""
|
|
320
|
+
public_flow_expiration: int = Field(default=86400, gt=600)
|
|
321
|
+
"""The time in seconds after which a public temporary flow will be considered expired and eligible for cleanup.
|
|
322
|
+
Default is 24 hours (86400 seconds). Minimum is 600 seconds (10 minutes)."""
|
|
323
|
+
event_delivery: Literal["polling", "streaming", "direct"] = "streaming"
|
|
324
|
+
"""How to deliver build events to the frontend. Can be 'polling', 'streaming' or 'direct'."""
|
|
325
|
+
lazy_load_components: bool = False
|
|
326
|
+
"""If set to True, Langflow will only partially load components at startup and fully load them on demand.
|
|
327
|
+
This significantly reduces startup time but may cause a slight delay when a component is first used."""
|
|
328
|
+
|
|
329
|
+
# Starter Projects
|
|
330
|
+
create_starter_projects: bool = True
|
|
331
|
+
"""If set to True, Langflow will create starter projects. If False, skips all starter project setup.
|
|
332
|
+
Note that this doesn't check if the starter projects are already loaded in the db;
|
|
333
|
+
this is intended to be used to skip all startup project logic."""
|
|
334
|
+
update_starter_projects: bool = True
|
|
335
|
+
"""If set to True, Langflow will update starter projects."""
|
|
336
|
+
|
|
337
|
+
# SSRF Protection
|
|
338
|
+
ssrf_protection_enabled: bool = False
|
|
339
|
+
"""If set to True, Langflow will enable SSRF (Server-Side Request Forgery) protection.
|
|
340
|
+
When enabled, blocks requests to private IP ranges, localhost, and cloud metadata endpoints.
|
|
341
|
+
When False (default), no URL validation is performed, allowing requests to any destination
|
|
342
|
+
including internal services, private networks, and cloud metadata endpoints.
|
|
343
|
+
Default is False for backward compatibility. In v2.0, this will be changed to True.
|
|
344
|
+
|
|
345
|
+
Note: When ssrf_protection_enabled is disabled, the ssrf_allowed_hosts setting is ignored and has no effect."""
|
|
346
|
+
ssrf_allowed_hosts: list[str] = []
|
|
347
|
+
"""Comma-separated list of hosts/IPs/CIDR ranges to allow despite SSRF protection.
|
|
348
|
+
Examples: 'internal-api.company.local,192.168.1.0/24,10.0.0.5,*.dev.internal'
|
|
349
|
+
Supports exact hostnames, wildcard domains (*.example.com), exact IPs, and CIDR ranges.
|
|
350
|
+
|
|
351
|
+
Note: This setting only takes effect when ssrf_protection_enabled is True.
|
|
352
|
+
When protection is disabled, all hosts are allowed regardless of this setting."""
|
|
353
|
+
|
|
354
|
+
@field_validator("cors_origins", mode="before")
|
|
355
|
+
@classmethod
|
|
356
|
+
def validate_cors_origins(cls, value):
|
|
357
|
+
"""Convert comma-separated string to list if needed."""
|
|
358
|
+
if isinstance(value, str) and value != "*":
|
|
359
|
+
if "," in value:
|
|
360
|
+
# Convert comma-separated string to list
|
|
361
|
+
return [origin.strip() for origin in value.split(",")]
|
|
362
|
+
# Convert single origin to list for consistency
|
|
363
|
+
return [value]
|
|
364
|
+
return value
|
|
365
|
+
|
|
366
|
+
@field_validator("use_noop_database", mode="before")
|
|
367
|
+
@classmethod
|
|
368
|
+
def set_use_noop_database(cls, value):
|
|
369
|
+
if value:
|
|
370
|
+
logger.info("Running with NOOP database session. All DB operations are disabled.")
|
|
371
|
+
return value
|
|
372
|
+
|
|
373
|
+
@field_validator("event_delivery", mode="before")
|
|
374
|
+
@classmethod
|
|
375
|
+
def set_event_delivery(cls, value, info):
|
|
376
|
+
# If workers > 1, we need to use direct delivery
|
|
377
|
+
# because polling and streaming are not supported
|
|
378
|
+
# in multi-worker environments
|
|
379
|
+
if info.data.get("workers", 1) > 1:
|
|
380
|
+
logger.warning("Multi-worker environment detected, using direct event delivery")
|
|
381
|
+
return "direct"
|
|
382
|
+
return value
|
|
383
|
+
|
|
384
|
+
@field_validator("user_agent", mode="after")
|
|
385
|
+
@classmethod
|
|
386
|
+
def set_user_agent(cls, value):
|
|
387
|
+
if not value:
|
|
388
|
+
value = "Langflow"
|
|
389
|
+
import os
|
|
390
|
+
|
|
391
|
+
os.environ["USER_AGENT"] = value
|
|
392
|
+
logger.debug(f"Setting user agent to {value}")
|
|
393
|
+
return value
|
|
394
|
+
|
|
395
|
+
@field_validator("mcp_composer_version", mode="before")
|
|
396
|
+
@classmethod
|
|
397
|
+
def validate_mcp_composer_version(cls, value):
|
|
398
|
+
"""Ensure the version string has a version specifier prefix.
|
|
399
|
+
|
|
400
|
+
If a bare version like '0.1.0.7' is provided, prepend '~=' to allow patch updates.
|
|
401
|
+
Supports PEP 440 specifiers: ==, !=, <=, >=, <, >, ~=, ===
|
|
402
|
+
"""
|
|
403
|
+
if not value:
|
|
404
|
+
return "==0.1.0.8.10" # Default
|
|
405
|
+
|
|
406
|
+
# Check if it already has a version specifier
|
|
407
|
+
# Order matters: check longer specifiers first to avoid false matches
|
|
408
|
+
specifiers = ["===", "==", "!=", "<=", ">=", "~=", "<", ">"]
|
|
409
|
+
if any(value.startswith(spec) for spec in specifiers):
|
|
410
|
+
return value
|
|
411
|
+
|
|
412
|
+
# If it's a bare version number, add ~= prefix
|
|
413
|
+
# This regex matches version numbers like 0.1.0.7, 1.2.3, etc.
|
|
414
|
+
import re
|
|
415
|
+
|
|
416
|
+
if re.match(r"^\d+(\.\d+)*", value):
|
|
417
|
+
logger.debug(f"Adding ~= prefix to bare version '{value}' -> '~={value}'")
|
|
418
|
+
return f"~={value}"
|
|
419
|
+
|
|
420
|
+
# If we can't determine, return as-is and let uvx handle it
|
|
421
|
+
return value
|
|
422
|
+
|
|
423
|
+
@field_validator("variables_to_get_from_environment", mode="before")
|
|
424
|
+
@classmethod
|
|
425
|
+
def set_variables_to_get_from_environment(cls, value):
|
|
426
|
+
import os
|
|
427
|
+
|
|
428
|
+
if isinstance(value, str):
|
|
429
|
+
value = value.split(",")
|
|
430
|
+
|
|
431
|
+
result = list(set(VARIABLES_TO_GET_FROM_ENVIRONMENT + value))
|
|
432
|
+
|
|
433
|
+
# Add agentic variables if agentic_experience is enabled
|
|
434
|
+
# Check env var directly since we can't access instance attributes in validator
|
|
435
|
+
if os.getenv("LANGFLOW_AGENTIC_EXPERIENCE", "true").lower() == "true":
|
|
436
|
+
result.extend(AGENTIC_VARIABLES)
|
|
437
|
+
|
|
438
|
+
return list(set(result))
|
|
439
|
+
|
|
440
|
+
@field_validator("log_file", mode="before")
|
|
441
|
+
@classmethod
|
|
442
|
+
def set_log_file(cls, value):
|
|
443
|
+
if isinstance(value, Path):
|
|
444
|
+
value = str(value)
|
|
445
|
+
return value
|
|
446
|
+
|
|
447
|
+
@field_validator("config_dir", mode="before")
|
|
448
|
+
@classmethod
|
|
449
|
+
def set_langflow_dir(cls, value):
|
|
450
|
+
if not value:
|
|
451
|
+
from platformdirs import user_cache_dir
|
|
452
|
+
|
|
453
|
+
# Define the app name and author
|
|
454
|
+
app_name = "langflow"
|
|
455
|
+
app_author = "langflow"
|
|
456
|
+
|
|
457
|
+
# Get the cache directory for the application
|
|
458
|
+
cache_dir = user_cache_dir(app_name, app_author)
|
|
459
|
+
|
|
460
|
+
# Create a .langflow directory inside the cache directory
|
|
461
|
+
value = Path(cache_dir)
|
|
462
|
+
value.mkdir(parents=True, exist_ok=True)
|
|
463
|
+
|
|
464
|
+
if isinstance(value, str):
|
|
465
|
+
value = Path(value)
|
|
466
|
+
# Resolve to absolute path to handle relative paths correctly
|
|
467
|
+
value = value.resolve()
|
|
468
|
+
if not value.exists():
|
|
469
|
+
value.mkdir(parents=True, exist_ok=True)
|
|
470
|
+
|
|
471
|
+
return str(value)
|
|
472
|
+
|
|
473
|
+
@field_validator("database_url", mode="before")
|
|
474
|
+
@classmethod
|
|
475
|
+
def set_database_url(cls, value, info):
|
|
476
|
+
if value and not is_valid_database_url(value):
|
|
477
|
+
msg = f"Invalid database_url provided: '{value}'"
|
|
478
|
+
raise ValueError(msg)
|
|
479
|
+
|
|
480
|
+
logger.debug("No database_url provided, trying LANGFLOW_DATABASE_URL env variable")
|
|
481
|
+
if langflow_database_url := os.getenv("LANGFLOW_DATABASE_URL"):
|
|
482
|
+
value = langflow_database_url
|
|
483
|
+
logger.debug("Using LANGFLOW_DATABASE_URL env variable.")
|
|
484
|
+
else:
|
|
485
|
+
logger.debug("No database_url env variable, using sqlite database")
|
|
486
|
+
# Originally, we used sqlite:///./langflow.db
|
|
487
|
+
# so we need to migrate to the new format
|
|
488
|
+
# if there is a database in that location
|
|
489
|
+
if not info.data["config_dir"]:
|
|
490
|
+
msg = "config_dir not set, please set it or provide a database_url"
|
|
491
|
+
raise ValueError(msg)
|
|
492
|
+
|
|
493
|
+
from lfx.utils.version import get_version_info
|
|
494
|
+
from lfx.utils.version import is_pre_release as langflow_is_pre_release
|
|
495
|
+
|
|
496
|
+
version = get_version_info()["version"]
|
|
497
|
+
is_pre_release = langflow_is_pre_release(version)
|
|
498
|
+
|
|
499
|
+
if info.data["save_db_in_config_dir"]:
|
|
500
|
+
database_dir = info.data["config_dir"]
|
|
501
|
+
logger.debug(f"Saving database to config_dir: {database_dir}")
|
|
502
|
+
else:
|
|
503
|
+
database_dir = Path(__file__).parent.parent.parent.resolve()
|
|
504
|
+
logger.debug(f"Saving database to langflow directory: {database_dir}")
|
|
505
|
+
|
|
506
|
+
pre_db_file_name = "langflow-pre.db"
|
|
507
|
+
db_file_name = "langflow.db"
|
|
508
|
+
new_pre_path = f"{database_dir}/{pre_db_file_name}"
|
|
509
|
+
new_path = f"{database_dir}/{db_file_name}"
|
|
510
|
+
final_path = None
|
|
511
|
+
if is_pre_release:
|
|
512
|
+
if Path(new_pre_path).exists():
|
|
513
|
+
final_path = new_pre_path
|
|
514
|
+
elif Path(new_path).exists() and info.data["save_db_in_config_dir"]:
|
|
515
|
+
# We need to copy the current db to the new location
|
|
516
|
+
logger.debug("Copying existing database to new location")
|
|
517
|
+
copy2(new_path, new_pre_path)
|
|
518
|
+
logger.debug(f"Copied existing database to {new_pre_path}")
|
|
519
|
+
elif Path(f"./{db_file_name}").exists() and info.data["save_db_in_config_dir"]:
|
|
520
|
+
logger.debug("Copying existing database to new location")
|
|
521
|
+
copy2(f"./{db_file_name}", new_pre_path)
|
|
522
|
+
logger.debug(f"Copied existing database to {new_pre_path}")
|
|
523
|
+
else:
|
|
524
|
+
logger.debug(f"Creating new database at {new_pre_path}")
|
|
525
|
+
final_path = new_pre_path
|
|
526
|
+
elif Path(new_path).exists():
|
|
527
|
+
logger.debug(f"Database already exists at {new_path}, using it")
|
|
528
|
+
final_path = new_path
|
|
529
|
+
elif Path(f"./{db_file_name}").exists():
|
|
530
|
+
try:
|
|
531
|
+
logger.debug("Copying existing database to new location")
|
|
532
|
+
copy2(f"./{db_file_name}", new_path)
|
|
533
|
+
logger.debug(f"Copied existing database to {new_path}")
|
|
534
|
+
except OSError:
|
|
535
|
+
logger.exception("Failed to copy database, using default path")
|
|
536
|
+
new_path = f"./{db_file_name}"
|
|
537
|
+
else:
|
|
538
|
+
final_path = new_path
|
|
539
|
+
|
|
540
|
+
if final_path is None:
|
|
541
|
+
final_path = new_pre_path if is_pre_release else new_path
|
|
542
|
+
|
|
543
|
+
value = f"sqlite:///{final_path}"
|
|
544
|
+
|
|
545
|
+
return value
|
|
546
|
+
|
|
547
|
+
@field_validator("components_path", mode="before")
|
|
548
|
+
@classmethod
|
|
549
|
+
def set_components_path(cls, value):
|
|
550
|
+
"""Processes and updates the components path list, incorporating environment variable overrides.
|
|
551
|
+
|
|
552
|
+
If the `LANGFLOW_COMPONENTS_PATH` environment variable is set and points to an existing path, it is
|
|
553
|
+
appended to the provided list if not already present. If the input list is empty or missing, it is
|
|
554
|
+
set to an empty list.
|
|
555
|
+
"""
|
|
556
|
+
if os.getenv("LANGFLOW_COMPONENTS_PATH"):
|
|
557
|
+
logger.debug("Adding LANGFLOW_COMPONENTS_PATH to components_path")
|
|
558
|
+
langflow_component_path = os.getenv("LANGFLOW_COMPONENTS_PATH")
|
|
559
|
+
if Path(langflow_component_path).exists() and langflow_component_path not in value:
|
|
560
|
+
if isinstance(langflow_component_path, list):
|
|
561
|
+
for path in langflow_component_path:
|
|
562
|
+
if path not in value:
|
|
563
|
+
value.append(path)
|
|
564
|
+
logger.debug(f"Extending {langflow_component_path} to components_path")
|
|
565
|
+
elif langflow_component_path not in value:
|
|
566
|
+
value.append(langflow_component_path)
|
|
567
|
+
logger.debug(f"Appending {langflow_component_path} to components_path")
|
|
568
|
+
|
|
569
|
+
if not value:
|
|
570
|
+
value = [BASE_COMPONENTS_PATH]
|
|
571
|
+
logger.debug("Setting default components path to components_path")
|
|
572
|
+
else:
|
|
573
|
+
if isinstance(value, Path):
|
|
574
|
+
value = [str(value)]
|
|
575
|
+
elif isinstance(value, list):
|
|
576
|
+
value = [str(p) if isinstance(p, Path) else p for p in value]
|
|
577
|
+
logger.debug("Adding default components path to components_path")
|
|
578
|
+
|
|
579
|
+
logger.debug(f"Components path: {value}")
|
|
580
|
+
return value
|
|
581
|
+
|
|
582
|
+
model_config = SettingsConfigDict(validate_assignment=True, extra="ignore", env_prefix="LANGFLOW_")
|
|
583
|
+
|
|
584
|
+
async def update_from_yaml(self, file_path: str, *, dev: bool = False) -> None:
|
|
585
|
+
new_settings = await load_settings_from_yaml(file_path)
|
|
586
|
+
self.components_path = new_settings.components_path or []
|
|
587
|
+
self.dev = dev
|
|
588
|
+
|
|
589
|
+
def update_settings(self, **kwargs) -> None:
|
|
590
|
+
logger.debug("Updating settings")
|
|
591
|
+
for key, value in kwargs.items():
|
|
592
|
+
# value may contain sensitive information, so we don't want to log it
|
|
593
|
+
if not hasattr(self, key):
|
|
594
|
+
logger.debug(f"Key {key} not found in settings")
|
|
595
|
+
continue
|
|
596
|
+
logger.debug(f"Updating {key}")
|
|
597
|
+
if isinstance(getattr(self, key), list):
|
|
598
|
+
# value might be a '[something]' string
|
|
599
|
+
value_ = value
|
|
600
|
+
with contextlib.suppress(json.decoder.JSONDecodeError):
|
|
601
|
+
value_ = orjson.loads(str(value))
|
|
602
|
+
if isinstance(value_, list):
|
|
603
|
+
for item in value_:
|
|
604
|
+
item_ = str(item) if isinstance(item, Path) else item
|
|
605
|
+
if item_ not in getattr(self, key):
|
|
606
|
+
getattr(self, key).append(item_)
|
|
607
|
+
logger.debug(f"Extended {key}")
|
|
608
|
+
else:
|
|
609
|
+
value_ = str(value_) if isinstance(value_, Path) else value_
|
|
610
|
+
if value_ not in getattr(self, key):
|
|
611
|
+
getattr(self, key).append(value_)
|
|
612
|
+
logger.debug(f"Appended {key}")
|
|
613
|
+
|
|
614
|
+
else:
|
|
615
|
+
setattr(self, key, value)
|
|
616
|
+
logger.debug(f"Updated {key}")
|
|
617
|
+
logger.debug(f"{key}: {getattr(self, key)}")
|
|
618
|
+
|
|
619
|
+
@property
|
|
620
|
+
def voice_mode_available(self) -> bool:
|
|
621
|
+
"""Check if voice mode is available by testing webrtcvad import."""
|
|
622
|
+
try:
|
|
623
|
+
import webrtcvad # noqa: F401
|
|
624
|
+
except ImportError:
|
|
625
|
+
return False
|
|
626
|
+
else:
|
|
627
|
+
return True
|
|
628
|
+
|
|
629
|
+
@classmethod
|
|
630
|
+
@override
|
|
631
|
+
def settings_customise_sources( # type: ignore[misc]
|
|
632
|
+
cls,
|
|
633
|
+
settings_cls: type[BaseSettings],
|
|
634
|
+
init_settings: PydanticBaseSettingsSource,
|
|
635
|
+
env_settings: PydanticBaseSettingsSource,
|
|
636
|
+
dotenv_settings: PydanticBaseSettingsSource,
|
|
637
|
+
file_secret_settings: PydanticBaseSettingsSource,
|
|
638
|
+
) -> tuple[PydanticBaseSettingsSource, ...]:
|
|
639
|
+
return (CustomSource(settings_cls),)
|
|
640
|
+
|
|
641
|
+
|
|
642
|
+
def save_settings_to_yaml(settings: Settings, file_path: str) -> None:
|
|
643
|
+
with Path(file_path).open("w", encoding="utf-8") as f:
|
|
644
|
+
settings_dict = settings.model_dump()
|
|
645
|
+
yaml.dump(settings_dict, f)
|
|
646
|
+
|
|
647
|
+
|
|
648
|
+
async def load_settings_from_yaml(file_path: str) -> Settings:
|
|
649
|
+
# Check if a string is a valid path or a file name
|
|
650
|
+
if "/" not in file_path:
|
|
651
|
+
# Get current path
|
|
652
|
+
current_path = Path(__file__).resolve().parent
|
|
653
|
+
file_path_ = Path(current_path) / file_path
|
|
654
|
+
else:
|
|
655
|
+
file_path_ = Path(file_path)
|
|
656
|
+
|
|
657
|
+
async with async_open(file_path_.name, encoding="utf-8") as f:
|
|
658
|
+
content = await f.read()
|
|
659
|
+
settings_dict = yaml.safe_load(content)
|
|
660
|
+
settings_dict = {k.upper(): v for k, v in settings_dict.items()}
|
|
661
|
+
|
|
662
|
+
for key in settings_dict:
|
|
663
|
+
if key not in Settings.model_fields:
|
|
664
|
+
msg = f"Key {key} not found in settings"
|
|
665
|
+
raise KeyError(msg)
|
|
666
|
+
await logger.adebug(f"Loading {len(settings_dict[key])} {key} from {file_path}")
|
|
667
|
+
|
|
668
|
+
return await asyncio.to_thread(Settings, **settings_dict)
|