lfx-nightly 0.1.11.dev0__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.
- lfx/__init__.py +0 -0
- lfx/__main__.py +25 -0
- lfx/base/__init__.py +0 -0
- lfx/base/agents/__init__.py +0 -0
- lfx/base/agents/agent.py +268 -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 +346 -0
- lfx/base/agents/utils.py +205 -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 +1291 -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 +685 -0
- lfx/base/data/docling_utils.py +245 -0
- lfx/base/data/utils.py +198 -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/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 +20 -0
- lfx/base/io/text.py +22 -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 +1398 -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 +47 -0
- lfx/base/models/aws_constants.py +151 -0
- lfx/base/models/chat_result.py +76 -0
- lfx/base/models/google_generative_ai_constants.py +70 -0
- lfx/base/models/groq_constants.py +134 -0
- lfx/base/models/model.py +375 -0
- lfx/base/models/model_input_constants.py +307 -0
- lfx/base/models/model_metadata.py +41 -0
- lfx/base/models/model_utils.py +8 -0
- lfx/base/models/novita_constants.py +35 -0
- lfx/base/models/ollama_constants.py +49 -0
- lfx/base/models/openai_constants.py +122 -0
- lfx/base/models/sambanova_constants.py +18 -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 +224 -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 +319 -0
- lfx/cli/common.py +650 -0
- lfx/cli/run.py +441 -0
- lfx/cli/script_loader.py +247 -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 +411 -0
- lfx/components/_importing.py +42 -0
- lfx/components/agentql/__init__.py +3 -0
- lfx/components/agentql/agentql_api.py +151 -0
- lfx/components/agents/__init__.py +34 -0
- lfx/components/agents/agent.py +558 -0
- lfx/components/agents/mcp_component.py +501 -0
- lfx/components/aiml/__init__.py +37 -0
- lfx/components/aiml/aiml.py +112 -0
- lfx/components/aiml/aiml_embeddings.py +37 -0
- lfx/components/amazon/__init__.py +36 -0
- lfx/components/amazon/amazon_bedrock_embedding.py +109 -0
- lfx/components/amazon/amazon_bedrock_model.py +124 -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 +163 -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 +167 -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/composio/__init__.py +74 -0
- lfx/components/composio/composio_api.py +268 -0
- lfx/components/composio/dropbox_compnent.py +11 -0
- lfx/components/composio/github_composio.py +11 -0
- lfx/components/composio/gmail_composio.py +38 -0
- lfx/components/composio/googlecalendar_composio.py +11 -0
- lfx/components/composio/googlemeet_composio.py +11 -0
- lfx/components/composio/googletasks_composio.py +8 -0
- lfx/components/composio/linear_composio.py +11 -0
- lfx/components/composio/outlook_composio.py +11 -0
- lfx/components/composio/reddit_composio.py +11 -0
- lfx/components/composio/slack_composio.py +582 -0
- lfx/components/composio/slackbot_composio.py +11 -0
- lfx/components/composio/supabase_composio.py +11 -0
- lfx/components/composio/todoist_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 +107 -0
- lfx/components/crewai/hierarchical_crew.py +46 -0
- lfx/components/crewai/hierarchical_task.py +44 -0
- lfx/components/crewai/sequential_crew.py +52 -0
- lfx/components/crewai/sequential_task.py +73 -0
- lfx/components/crewai/sequential_task_agent.py +143 -0
- lfx/components/custom_component/__init__.py +34 -0
- lfx/components/custom_component/custom_component.py +31 -0
- lfx/components/data/__init__.py +64 -0
- lfx/components/data/api_request.py +544 -0
- lfx/components/data/csv_to_data.py +95 -0
- lfx/components/data/directory.py +113 -0
- lfx/components/data/file.py +577 -0
- lfx/components/data/json_to_data.py +98 -0
- lfx/components/data/news_search.py +164 -0
- lfx/components/data/rss.py +69 -0
- lfx/components/data/sql_executor.py +101 -0
- lfx/components/data/url.py +311 -0
- lfx/components/data/web_search.py +112 -0
- lfx/components/data/webhook.py +56 -0
- lfx/components/datastax/__init__.py +70 -0
- lfx/components/datastax/astra_assistant_manager.py +306 -0
- lfx/components/datastax/astra_db.py +75 -0
- lfx/components/datastax/astra_vectorize.py +124 -0
- lfx/components/datastax/astradb.py +1285 -0
- lfx/components/datastax/astradb_cql.py +314 -0
- lfx/components/datastax/astradb_graph.py +330 -0
- lfx/components/datastax/astradb_tool.py +414 -0
- lfx/components/datastax/astradb_vectorstore.py +1285 -0
- lfx/components/datastax/cassandra.py +92 -0
- lfx/components/datastax/create_assistant.py +58 -0
- lfx/components/datastax/create_thread.py +32 -0
- lfx/components/datastax/dotenv.py +35 -0
- lfx/components/datastax/get_assistant.py +37 -0
- lfx/components/datastax/getenvvar.py +30 -0
- lfx/components/datastax/graph_rag.py +141 -0
- lfx/components/datastax/hcd.py +314 -0
- lfx/components/datastax/list_assistants.py +25 -0
- lfx/components/datastax/run.py +89 -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 +231 -0
- lfx/components/docling/docling_remote.py +193 -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 +243 -0
- lfx/components/embeddings/__init__.py +37 -0
- lfx/components/embeddings/similarity.py +76 -0
- lfx/components/embeddings/text_embedder.py +64 -0
- lfx/components/exa/__init__.py +3 -0
- lfx/components/exa/exa_search.py +68 -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/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 +192 -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 +147 -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 +136 -0
- lfx/components/helpers/__init__.py +52 -0
- lfx/components/helpers/calculator_core.py +89 -0
- lfx/components/helpers/create_list.py +40 -0
- lfx/components/helpers/current_date.py +42 -0
- lfx/components/helpers/id_generator.py +42 -0
- lfx/components/helpers/memory.py +251 -0
- lfx/components/helpers/output_parser.py +45 -0
- lfx/components/helpers/store_message.py +90 -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 +197 -0
- lfx/components/huggingface/huggingface_inference_api.py +106 -0
- lfx/components/ibm/__init__.py +34 -0
- lfx/components/ibm/watsonx.py +203 -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 +38 -0
- lfx/components/input_output/chat.py +120 -0
- lfx/components/input_output/chat_output.py +200 -0
- lfx/components/input_output/text.py +27 -0
- lfx/components/input_output/text_output.py +29 -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/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 +107 -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 +45 -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/lmstudio/__init__.py +34 -0
- lfx/components/lmstudio/lmstudioembeddings.py +89 -0
- lfx/components/lmstudio/lmstudiomodel.py +129 -0
- lfx/components/logic/__init__.py +52 -0
- lfx/components/logic/conditional_router.py +171 -0
- lfx/components/logic/data_conditional_router.py +125 -0
- lfx/components/logic/flow_tool.py +110 -0
- lfx/components/logic/listen.py +29 -0
- lfx/components/logic/loop.py +125 -0
- lfx/components/logic/notify.py +88 -0
- lfx/components/logic/pass_message.py +35 -0
- lfx/components/logic/run_flow.py +71 -0
- lfx/components/logic/sub_flow.py +114 -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 +136 -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 +34 -0
- lfx/components/models/embedding_model.py +114 -0
- lfx/components/models/language_model.py +144 -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 +157 -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 +330 -0
- lfx/components/ollama/ollama_embeddings.py +106 -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 +202 -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 +117 -0
- lfx/components/processing/alter_metadata.py +108 -0
- lfx/components/processing/batch_run.py +205 -0
- lfx/components/processing/combine_text.py +39 -0
- lfx/components/processing/converter.py +159 -0
- lfx/components/processing/create_data.py +110 -0
- lfx/components/processing/data_operations.py +438 -0
- lfx/components/processing/data_to_dataframe.py +70 -0
- lfx/components/processing/dataframe_operations.py +313 -0
- lfx/components/processing/extract_key.py +53 -0
- lfx/components/processing/filter_data.py +42 -0
- lfx/components/processing/filter_data_values.py +88 -0
- lfx/components/processing/json_cleaner.py +103 -0
- lfx/components/processing/lambda_filter.py +154 -0
- lfx/components/processing/llm_router.py +499 -0
- lfx/components/processing/merge_data.py +90 -0
- lfx/components/processing/message_to_data.py +36 -0
- lfx/components/processing/parse_data.py +70 -0
- lfx/components/processing/parse_dataframe.py +68 -0
- lfx/components/processing/parse_json_data.py +90 -0
- lfx/components/processing/parser.py +143 -0
- lfx/components/processing/prompt.py +67 -0
- lfx/components/processing/python_repl_core.py +98 -0
- lfx/components/processing/regex.py +82 -0
- lfx/components/processing/save_file.py +225 -0
- lfx/components/processing/select_data.py +48 -0
- lfx/components/processing/split_text.py +141 -0
- lfx/components/processing/structured_output.py +202 -0
- lfx/components/processing/update_data.py +160 -0
- lfx/components/prototypes/__init__.py +34 -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 +72 -0
- lfx/components/tools/calculator.py +108 -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 +327 -0
- lfx/components/tools/python_repl.py +97 -0
- lfx/components/tools/search_api.py +87 -0
- lfx/components/tools/searxng.py +145 -0
- lfx/components/tools/serp_api.py +119 -0
- lfx/components/tools/tavily_search_tool.py +344 -0
- lfx/components/tools/wikidata_api.py +102 -0
- lfx/components/tools/wikipedia_api.py +49 -0
- lfx/components/tools/yahoo_finance.py +129 -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 +291 -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 +179 -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/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 +40 -0
- lfx/components/vectorstores/astradb.py +1285 -0
- lfx/components/vectorstores/astradb_graph.py +319 -0
- lfx/components/vectorstores/cassandra.py +264 -0
- lfx/components/vectorstores/cassandra_graph.py +238 -0
- lfx/components/vectorstores/chroma.py +167 -0
- lfx/components/vectorstores/clickhouse.py +135 -0
- lfx/components/vectorstores/couchbase.py +102 -0
- lfx/components/vectorstores/elasticsearch.py +267 -0
- lfx/components/vectorstores/faiss.py +111 -0
- lfx/components/vectorstores/graph_rag.py +141 -0
- lfx/components/vectorstores/hcd.py +314 -0
- lfx/components/vectorstores/local_db.py +261 -0
- lfx/components/vectorstores/milvus.py +115 -0
- lfx/components/vectorstores/mongodb_atlas.py +213 -0
- lfx/components/vectorstores/opensearch.py +243 -0
- lfx/components/vectorstores/pgvector.py +72 -0
- lfx/components/vectorstores/pinecone.py +134 -0
- lfx/components/vectorstores/qdrant.py +109 -0
- lfx/components/vectorstores/supabase.py +76 -0
- lfx/components/vectorstores/upstash.py +124 -0
- lfx/components/vectorstores/vectara.py +97 -0
- lfx/components/vectorstores/vectara_rag.py +164 -0
- lfx/components/vectorstores/weaviate.py +89 -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/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 +118 -0
- lfx/components/zep/__init__.py +3 -0
- lfx/components/zep/zep.py +44 -0
- lfx/constants.py +6 -0
- lfx/custom/__init__.py +7 -0
- lfx/custom/attributes.py +86 -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 +1808 -0
- lfx/custom/custom_component/component_with_cache.py +8 -0
- lfx/custom/custom_component/custom_component.py +588 -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 +488 -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 +215 -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 +277 -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 +2238 -0
- lfx/graph/graph/constants.py +63 -0
- lfx/graph/graph/runnable_vertices_manager.py +133 -0
- lfx/graph/graph/schema.py +52 -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 +237 -0
- lfx/graph/utils.py +200 -0
- lfx/graph/vertex/__init__.py +0 -0
- lfx/graph/vertex/base.py +823 -0
- lfx/graph/vertex/constants.py +0 -0
- lfx/graph/vertex/exceptions.py +4 -0
- lfx/graph/vertex/param_handler.py +264 -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 +1 -0
- lfx/helpers/base_model.py +71 -0
- lfx/helpers/custom.py +13 -0
- lfx/helpers/data.py +167 -0
- lfx/helpers/flow.py +194 -0
- lfx/inputs/__init__.py +68 -0
- lfx/inputs/constants.py +2 -0
- lfx/inputs/input_mixin.py +328 -0
- lfx/inputs/inputs.py +714 -0
- lfx/inputs/validators.py +19 -0
- lfx/interface/__init__.py +6 -0
- lfx/interface/components.py +489 -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 +224 -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 +289 -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 +385 -0
- lfx/memory/__init__.py +90 -0
- lfx/memory/stubs.py +283 -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/data.py +308 -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 +131 -0
- lfx/schema/json_schema.py +141 -0
- lfx/schema/log.py +61 -0
- lfx/schema/message.py +473 -0
- lfx/schema/openai_responses_schemas.py +74 -0
- lfx/schema/properties.py +41 -0
- lfx/schema/schema.py +171 -0
- lfx/schema/serialize.py +13 -0
- lfx/schema/table.py +140 -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 +23 -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/deps.py +129 -0
- lfx/services/factory.py +19 -0
- lfx/services/initialize.py +19 -0
- lfx/services/interfaces.py +103 -0
- lfx/services/manager.py +172 -0
- lfx/services/schema.py +20 -0
- lfx/services/session.py +82 -0
- lfx/services/settings/__init__.py +3 -0
- lfx/services/settings/auth.py +130 -0
- lfx/services/settings/base.py +539 -0
- lfx/services/settings/constants.py +31 -0
- lfx/services/settings/factory.py +23 -0
- lfx/services/settings/feature_flags.py +12 -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 +155 -0
- lfx/services/storage/service.py +54 -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 +257 -0
- lfx/template/field/prompt.py +15 -0
- lfx/template/frontend_node/__init__.py +6 -0
- lfx/template/frontend_node/base.py +212 -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 +205 -0
- lfx/utils/data_structure.py +212 -0
- lfx/utils/exceptions.py +22 -0
- lfx/utils/helpers.py +28 -0
- lfx/utils/image.py +73 -0
- lfx/utils/lazy_load.py +15 -0
- lfx/utils/request_utils.py +18 -0
- lfx/utils/schemas.py +139 -0
- lfx/utils/util.py +481 -0
- lfx/utils/util_strings.py +56 -0
- lfx/utils/version.py +24 -0
- lfx_nightly-0.1.11.dev0.dist-info/METADATA +293 -0
- lfx_nightly-0.1.11.dev0.dist-info/RECORD +699 -0
- lfx_nightly-0.1.11.dev0.dist-info/WHEEL +4 -0
- lfx_nightly-0.1.11.dev0.dist-info/entry_points.txt +2 -0
@@ -0,0 +1,361 @@
|
|
1
|
+
import ast
|
2
|
+
import contextlib
|
3
|
+
import inspect
|
4
|
+
import traceback
|
5
|
+
from itertools import starmap
|
6
|
+
from pathlib import Path
|
7
|
+
from typing import Any
|
8
|
+
|
9
|
+
from cachetools import TTLCache, keys
|
10
|
+
from fastapi import HTTPException
|
11
|
+
|
12
|
+
from lfx.custom.eval import eval_custom_component_code
|
13
|
+
from lfx.custom.schema import CallableCodeDetails, ClassCodeDetails, MissingDefault
|
14
|
+
from lfx.log.logger import logger
|
15
|
+
|
16
|
+
|
17
|
+
class CodeSyntaxError(HTTPException):
|
18
|
+
pass
|
19
|
+
|
20
|
+
|
21
|
+
def get_data_type():
|
22
|
+
from lfx.schema.data import Data
|
23
|
+
|
24
|
+
return Data
|
25
|
+
|
26
|
+
|
27
|
+
def find_class_ast_node(class_obj):
|
28
|
+
"""Finds the AST node corresponding to the given class object."""
|
29
|
+
# Get the source file where the class is defined
|
30
|
+
source_file = inspect.getsourcefile(class_obj)
|
31
|
+
if not source_file:
|
32
|
+
return None, []
|
33
|
+
|
34
|
+
# Read the source code from the file
|
35
|
+
source_code = Path(source_file).read_text(encoding="utf-8")
|
36
|
+
|
37
|
+
# Parse the source code into an AST
|
38
|
+
tree = ast.parse(source_code)
|
39
|
+
|
40
|
+
# Search for the class definition node in the AST
|
41
|
+
class_node = None
|
42
|
+
import_nodes = []
|
43
|
+
for node in ast.walk(tree):
|
44
|
+
if isinstance(node, ast.ClassDef) and node.name == class_obj.__name__:
|
45
|
+
class_node = node
|
46
|
+
elif isinstance(node, ast.Import | ast.ImportFrom):
|
47
|
+
import_nodes.append(node)
|
48
|
+
|
49
|
+
return class_node, import_nodes
|
50
|
+
|
51
|
+
|
52
|
+
def imports_key(*args, **kwargs):
|
53
|
+
imports = kwargs.pop("imports")
|
54
|
+
key = keys.methodkey(*args, **kwargs)
|
55
|
+
key += tuple(imports)
|
56
|
+
return key
|
57
|
+
|
58
|
+
|
59
|
+
class CodeParser:
|
60
|
+
"""A parser for Python source code, extracting code details."""
|
61
|
+
|
62
|
+
def __init__(self, code: str | type) -> None:
|
63
|
+
"""Initializes the parser with the provided code."""
|
64
|
+
self.cache: TTLCache = TTLCache(maxsize=1024, ttl=60)
|
65
|
+
if isinstance(code, type):
|
66
|
+
if not inspect.isclass(code):
|
67
|
+
msg = "The provided code must be a class."
|
68
|
+
raise ValueError(msg)
|
69
|
+
# If the code is a class, get its source code
|
70
|
+
code = inspect.getsource(code)
|
71
|
+
self.code = code
|
72
|
+
self.data: dict[str, Any] = {
|
73
|
+
"imports": [],
|
74
|
+
"functions": [],
|
75
|
+
"classes": [],
|
76
|
+
"global_vars": [],
|
77
|
+
}
|
78
|
+
self.handlers = {
|
79
|
+
ast.Import: self.parse_imports,
|
80
|
+
ast.ImportFrom: self.parse_imports,
|
81
|
+
ast.FunctionDef: self.parse_functions,
|
82
|
+
ast.ClassDef: self.parse_classes,
|
83
|
+
ast.Assign: self.parse_global_vars,
|
84
|
+
}
|
85
|
+
|
86
|
+
def get_tree(self):
|
87
|
+
"""Parses the provided code to validate its syntax.
|
88
|
+
|
89
|
+
It tries to parse the code into an abstract syntax tree (AST).
|
90
|
+
"""
|
91
|
+
try:
|
92
|
+
tree = ast.parse(self.code)
|
93
|
+
except SyntaxError as err:
|
94
|
+
raise CodeSyntaxError(
|
95
|
+
status_code=400,
|
96
|
+
detail={"error": err.msg, "traceback": traceback.format_exc()},
|
97
|
+
) from err
|
98
|
+
|
99
|
+
return tree
|
100
|
+
|
101
|
+
def parse_node(self, node: ast.stmt | ast.AST) -> None:
|
102
|
+
"""Parses an AST node and updates the data dictionary with the relevant information."""
|
103
|
+
if handler := self.handlers.get(type(node)):
|
104
|
+
handler(node) # type: ignore[operator]
|
105
|
+
|
106
|
+
def parse_imports(self, node: ast.Import | ast.ImportFrom) -> None:
|
107
|
+
"""Extracts "imports" from the code, including aliases."""
|
108
|
+
if isinstance(node, ast.Import):
|
109
|
+
for alias in node.names:
|
110
|
+
if alias.asname:
|
111
|
+
self.data["imports"].append(f"{alias.name} as {alias.asname}")
|
112
|
+
else:
|
113
|
+
self.data["imports"].append(alias.name)
|
114
|
+
elif isinstance(node, ast.ImportFrom):
|
115
|
+
for alias in node.names:
|
116
|
+
if alias.asname:
|
117
|
+
self.data["imports"].append((node.module, f"{alias.name} as {alias.asname}"))
|
118
|
+
else:
|
119
|
+
self.data["imports"].append((node.module, alias.name))
|
120
|
+
|
121
|
+
def parse_functions(self, node: ast.FunctionDef) -> None:
|
122
|
+
"""Extracts "functions" from the code."""
|
123
|
+
self.data["functions"].append(self.parse_callable_details(node))
|
124
|
+
|
125
|
+
def parse_arg(self, arg, default):
|
126
|
+
"""Parses an argument and its default value."""
|
127
|
+
arg_dict = {"name": arg.arg, "default": default}
|
128
|
+
if arg.annotation:
|
129
|
+
arg_dict["type"] = ast.unparse(arg.annotation)
|
130
|
+
return arg_dict
|
131
|
+
|
132
|
+
# @cachedmethod(operator.attrgetter("cache"))
|
133
|
+
def construct_eval_env(self, return_type_str: str, imports) -> dict:
|
134
|
+
"""Constructs an evaluation environment.
|
135
|
+
|
136
|
+
Constructs an evaluation environment with the necessary imports for the return type,
|
137
|
+
taking into account module aliases.
|
138
|
+
"""
|
139
|
+
eval_env: dict = {}
|
140
|
+
for import_entry in imports:
|
141
|
+
if isinstance(import_entry, tuple): # from module import name
|
142
|
+
module, name = import_entry
|
143
|
+
if name in return_type_str:
|
144
|
+
exec(f"import {module}", eval_env)
|
145
|
+
exec(f"from {module} import {name}", eval_env)
|
146
|
+
else: # import module
|
147
|
+
module = import_entry
|
148
|
+
alias = None
|
149
|
+
if " as " in module:
|
150
|
+
module, alias = module.split(" as ")
|
151
|
+
if module in return_type_str or (alias and alias in return_type_str):
|
152
|
+
exec(f"import {module} as {alias or module}", eval_env)
|
153
|
+
return eval_env
|
154
|
+
|
155
|
+
def parse_callable_details(self, node: ast.FunctionDef) -> dict[str, Any]:
|
156
|
+
"""Extracts details from a single function or method node."""
|
157
|
+
return_type = None
|
158
|
+
if node.returns:
|
159
|
+
return_type_str = ast.unparse(node.returns)
|
160
|
+
eval_env = self.construct_eval_env(return_type_str, tuple(self.data["imports"]))
|
161
|
+
|
162
|
+
# Handle cases where the type is not found in the constructed environment
|
163
|
+
with contextlib.suppress(NameError):
|
164
|
+
return_type = eval(return_type_str, eval_env) # noqa: S307
|
165
|
+
|
166
|
+
func = CallableCodeDetails(
|
167
|
+
name=node.name,
|
168
|
+
doc=ast.get_docstring(node),
|
169
|
+
args=self.parse_function_args(node),
|
170
|
+
body=self.parse_function_body(node),
|
171
|
+
return_type=return_type,
|
172
|
+
has_return=self.parse_return_statement(node),
|
173
|
+
)
|
174
|
+
|
175
|
+
return func.model_dump()
|
176
|
+
|
177
|
+
def parse_function_args(self, node: ast.FunctionDef) -> list[dict[str, Any]]:
|
178
|
+
"""Parses the arguments of a function or method node."""
|
179
|
+
args = []
|
180
|
+
|
181
|
+
args += self.parse_positional_args(node)
|
182
|
+
args += self.parse_varargs(node)
|
183
|
+
args += self.parse_keyword_args(node)
|
184
|
+
# Commented out because we don't want kwargs
|
185
|
+
# showing up as fields in the frontend
|
186
|
+
args += self.parse_kwargs(node)
|
187
|
+
|
188
|
+
return args
|
189
|
+
|
190
|
+
def parse_positional_args(self, node: ast.FunctionDef) -> list[dict[str, Any]]:
|
191
|
+
"""Parses the positional arguments of a function or method node."""
|
192
|
+
num_args = len(node.args.args)
|
193
|
+
num_defaults = len(node.args.defaults)
|
194
|
+
num_missing_defaults = num_args - num_defaults
|
195
|
+
missing_defaults = [MissingDefault()] * num_missing_defaults
|
196
|
+
default_values = [ast.unparse(default).strip("'") if default else None for default in node.args.defaults]
|
197
|
+
# Now check all default values to see if there
|
198
|
+
# are any "None" values in the middle
|
199
|
+
default_values = [None if value == "None" else value for value in default_values]
|
200
|
+
|
201
|
+
defaults = missing_defaults + default_values
|
202
|
+
|
203
|
+
return list(starmap(self.parse_arg, zip(node.args.args, defaults, strict=True)))
|
204
|
+
|
205
|
+
def parse_varargs(self, node: ast.FunctionDef) -> list[dict[str, Any]]:
|
206
|
+
"""Parses the *args argument of a function or method node."""
|
207
|
+
args = []
|
208
|
+
|
209
|
+
if node.args.vararg:
|
210
|
+
args.append(self.parse_arg(node.args.vararg, None))
|
211
|
+
|
212
|
+
return args
|
213
|
+
|
214
|
+
def parse_keyword_args(self, node: ast.FunctionDef) -> list[dict[str, Any]]:
|
215
|
+
"""Parses the keyword-only arguments of a function or method node."""
|
216
|
+
kw_defaults = [None] * (len(node.args.kwonlyargs) - len(node.args.kw_defaults)) + [
|
217
|
+
ast.unparse(default) if default else None for default in node.args.kw_defaults
|
218
|
+
]
|
219
|
+
|
220
|
+
return list(starmap(self.parse_arg, zip(node.args.kwonlyargs, kw_defaults, strict=True)))
|
221
|
+
|
222
|
+
def parse_kwargs(self, node: ast.FunctionDef) -> list[dict[str, Any]]:
|
223
|
+
"""Parses the **kwargs argument of a function or method node."""
|
224
|
+
args = []
|
225
|
+
|
226
|
+
if node.args.kwarg:
|
227
|
+
args.append(self.parse_arg(node.args.kwarg, None))
|
228
|
+
|
229
|
+
return args
|
230
|
+
|
231
|
+
def parse_function_body(self, node: ast.FunctionDef) -> list[str]:
|
232
|
+
"""Parses the body of a function or method node."""
|
233
|
+
return [ast.unparse(line) for line in node.body]
|
234
|
+
|
235
|
+
def parse_return_statement(self, node: ast.FunctionDef) -> bool:
|
236
|
+
"""Parses the return statement of a function or method node, including nested returns."""
|
237
|
+
|
238
|
+
def has_return(node):
|
239
|
+
if isinstance(node, ast.Return):
|
240
|
+
return True
|
241
|
+
if isinstance(node, ast.If):
|
242
|
+
return any(has_return(child) for child in node.body) or any(has_return(child) for child in node.orelse)
|
243
|
+
if isinstance(node, ast.Try):
|
244
|
+
return (
|
245
|
+
any(has_return(child) for child in node.body)
|
246
|
+
or any(has_return(child) for child in node.handlers)
|
247
|
+
or any(has_return(child) for child in node.finalbody)
|
248
|
+
)
|
249
|
+
if isinstance(node, ast.For | ast.While):
|
250
|
+
return any(has_return(child) for child in node.body) or any(has_return(child) for child in node.orelse)
|
251
|
+
if isinstance(node, ast.With):
|
252
|
+
return any(has_return(child) for child in node.body)
|
253
|
+
return False
|
254
|
+
|
255
|
+
return any(has_return(child) for child in node.body)
|
256
|
+
|
257
|
+
def parse_assign(self, stmt):
|
258
|
+
"""Parses an Assign statement and returns a dictionary with the target's name and value."""
|
259
|
+
for target in stmt.targets:
|
260
|
+
if isinstance(target, ast.Name):
|
261
|
+
return {"name": target.id, "value": ast.unparse(stmt.value)}
|
262
|
+
return None
|
263
|
+
|
264
|
+
def parse_ann_assign(self, stmt):
|
265
|
+
"""Parses an AnnAssign statement and returns a dictionary with the target's name, value, and annotation."""
|
266
|
+
if isinstance(stmt.target, ast.Name):
|
267
|
+
return {
|
268
|
+
"name": stmt.target.id,
|
269
|
+
"value": ast.unparse(stmt.value) if stmt.value else None,
|
270
|
+
"annotation": ast.unparse(stmt.annotation),
|
271
|
+
}
|
272
|
+
return None
|
273
|
+
|
274
|
+
def parse_function_def(self, stmt):
|
275
|
+
"""Parse a FunctionDef statement.
|
276
|
+
|
277
|
+
Parse a FunctionDef statement and return the parsed method and a boolean indicating if it's an __init__ method.
|
278
|
+
"""
|
279
|
+
method = self.parse_callable_details(stmt)
|
280
|
+
return (method, True) if stmt.name == "__init__" else (method, False)
|
281
|
+
|
282
|
+
def get_base_classes(self):
|
283
|
+
"""Returns the base classes of the custom component class."""
|
284
|
+
try:
|
285
|
+
bases = self.execute_and_inspect_classes(self.code)
|
286
|
+
except Exception:
|
287
|
+
# If the code cannot be executed, return an empty list
|
288
|
+
bases = []
|
289
|
+
raise
|
290
|
+
return bases
|
291
|
+
|
292
|
+
def parse_classes(self, node: ast.ClassDef) -> None:
|
293
|
+
"""Extracts "classes" from the code, including inheritance and init methods."""
|
294
|
+
bases = self.get_base_classes()
|
295
|
+
nodes = []
|
296
|
+
for base in bases:
|
297
|
+
if base.__name__ == node.name or base.__name__ in {"CustomComponent", "Component", "BaseComponent"}:
|
298
|
+
continue
|
299
|
+
try:
|
300
|
+
class_node, import_nodes = find_class_ast_node(base)
|
301
|
+
if class_node is None:
|
302
|
+
continue
|
303
|
+
for import_node in import_nodes:
|
304
|
+
self.parse_imports(import_node)
|
305
|
+
nodes.append(class_node)
|
306
|
+
except Exception: # noqa: BLE001
|
307
|
+
logger.exception("Error finding base class node")
|
308
|
+
nodes.insert(0, node)
|
309
|
+
class_details = ClassCodeDetails(
|
310
|
+
name=node.name,
|
311
|
+
doc=ast.get_docstring(node),
|
312
|
+
bases=[b.__name__ for b in bases],
|
313
|
+
attributes=[],
|
314
|
+
methods=[],
|
315
|
+
init=None,
|
316
|
+
)
|
317
|
+
for _node in nodes:
|
318
|
+
self.process_class_node(_node, class_details)
|
319
|
+
self.data["classes"].append(class_details.model_dump())
|
320
|
+
|
321
|
+
def process_class_node(self, node, class_details) -> None:
|
322
|
+
for stmt in node.body:
|
323
|
+
if isinstance(stmt, ast.Assign):
|
324
|
+
if attr := self.parse_assign(stmt):
|
325
|
+
class_details.attributes.append(attr)
|
326
|
+
elif isinstance(stmt, ast.AnnAssign):
|
327
|
+
if attr := self.parse_ann_assign(stmt):
|
328
|
+
class_details.attributes.append(attr)
|
329
|
+
elif isinstance(stmt, ast.FunctionDef | ast.AsyncFunctionDef):
|
330
|
+
method, is_init = self.parse_function_def(stmt)
|
331
|
+
if is_init:
|
332
|
+
class_details.init = method
|
333
|
+
else:
|
334
|
+
class_details.methods.append(method)
|
335
|
+
|
336
|
+
def parse_global_vars(self, node: ast.Assign) -> None:
|
337
|
+
"""Extracts global variables from the code."""
|
338
|
+
global_var = {
|
339
|
+
"targets": [t.id if hasattr(t, "id") else ast.dump(t) for t in node.targets],
|
340
|
+
"value": ast.unparse(node.value),
|
341
|
+
}
|
342
|
+
self.data["global_vars"].append(global_var)
|
343
|
+
|
344
|
+
def execute_and_inspect_classes(self, code: str):
|
345
|
+
custom_component_class = eval_custom_component_code(code)
|
346
|
+
custom_component = custom_component_class(_code=code)
|
347
|
+
dunder_class = custom_component.__class__
|
348
|
+
# Get the base classes at two levels of inheritance
|
349
|
+
bases = []
|
350
|
+
for base in dunder_class.__bases__:
|
351
|
+
bases.append(base)
|
352
|
+
bases.extend(base.__bases__)
|
353
|
+
return bases
|
354
|
+
|
355
|
+
def parse_code(self) -> dict[str, Any]:
|
356
|
+
"""Runs all parsing operations and returns the resulting data."""
|
357
|
+
tree = self.get_tree()
|
358
|
+
|
359
|
+
for node in ast.walk(tree):
|
360
|
+
self.parse_node(node)
|
361
|
+
return self.data
|
File without changes
|
@@ -0,0 +1,128 @@
|
|
1
|
+
import copy
|
2
|
+
import operator
|
3
|
+
import re
|
4
|
+
from typing import TYPE_CHECKING, Any, ClassVar
|
5
|
+
|
6
|
+
from cachetools import TTLCache, cachedmethod
|
7
|
+
from fastapi import HTTPException
|
8
|
+
|
9
|
+
from lfx.custom import validate
|
10
|
+
from lfx.custom.attributes import ATTR_FUNC_MAPPING
|
11
|
+
from lfx.custom.code_parser.code_parser import CodeParser
|
12
|
+
from lfx.custom.eval import eval_custom_component_code
|
13
|
+
from lfx.log.logger import logger
|
14
|
+
|
15
|
+
if TYPE_CHECKING:
|
16
|
+
from uuid import UUID
|
17
|
+
|
18
|
+
|
19
|
+
class ComponentCodeNullError(HTTPException):
|
20
|
+
pass
|
21
|
+
|
22
|
+
|
23
|
+
class ComponentFunctionEntrypointNameNullError(HTTPException):
|
24
|
+
pass
|
25
|
+
|
26
|
+
|
27
|
+
class BaseComponent:
|
28
|
+
ERROR_CODE_NULL: ClassVar[str] = "Python code must be provided."
|
29
|
+
ERROR_FUNCTION_ENTRYPOINT_NAME_NULL: ClassVar[str] = "The name of the entrypoint function must be provided."
|
30
|
+
|
31
|
+
def __init__(self, **data) -> None:
|
32
|
+
self._code: str | None = None
|
33
|
+
self._function_entrypoint_name: str = "build"
|
34
|
+
self.field_config: dict = {}
|
35
|
+
self._user_id: str | UUID | None = None
|
36
|
+
self._template_config: dict = {}
|
37
|
+
|
38
|
+
self.cache: TTLCache = TTLCache(maxsize=1024, ttl=60)
|
39
|
+
|
40
|
+
for key, value in data.items():
|
41
|
+
if key == "user_id":
|
42
|
+
self._user_id = value
|
43
|
+
else:
|
44
|
+
setattr(self, key, value)
|
45
|
+
|
46
|
+
def __setattr__(self, key, value) -> None:
|
47
|
+
if key == "_user_id":
|
48
|
+
try:
|
49
|
+
if self._user_id is not None:
|
50
|
+
logger.warning("user_id is immutable and cannot be changed.")
|
51
|
+
except (KeyError, AttributeError):
|
52
|
+
pass
|
53
|
+
super().__setattr__(key, value)
|
54
|
+
|
55
|
+
@property
|
56
|
+
def code(self) -> str | None:
|
57
|
+
"""Get the component code."""
|
58
|
+
return self._code
|
59
|
+
|
60
|
+
@property
|
61
|
+
def function_entrypoint_name(self) -> str:
|
62
|
+
"""Get the function entrypoint name."""
|
63
|
+
return self._function_entrypoint_name
|
64
|
+
|
65
|
+
@cachedmethod(cache=operator.attrgetter("cache"))
|
66
|
+
def get_code_tree(self, code: str):
|
67
|
+
parser = CodeParser(code)
|
68
|
+
return parser.parse_code()
|
69
|
+
|
70
|
+
def get_function(self):
|
71
|
+
if not self._code:
|
72
|
+
raise ComponentCodeNullError(
|
73
|
+
status_code=400,
|
74
|
+
detail={"error": self.ERROR_CODE_NULL, "traceback": ""},
|
75
|
+
)
|
76
|
+
|
77
|
+
if not self._function_entrypoint_name:
|
78
|
+
raise ComponentFunctionEntrypointNameNullError(
|
79
|
+
status_code=400,
|
80
|
+
detail={
|
81
|
+
"error": self.ERROR_FUNCTION_ENTRYPOINT_NAME_NULL,
|
82
|
+
"traceback": "",
|
83
|
+
},
|
84
|
+
)
|
85
|
+
|
86
|
+
return validate.create_function(self._code, self._function_entrypoint_name)
|
87
|
+
|
88
|
+
@staticmethod
|
89
|
+
def get_template_config(component):
|
90
|
+
"""Gets the template configuration for the custom component itself."""
|
91
|
+
template_config = {}
|
92
|
+
|
93
|
+
for attribute, func in ATTR_FUNC_MAPPING.items():
|
94
|
+
if hasattr(component, attribute):
|
95
|
+
value = getattr(component, attribute)
|
96
|
+
if value is not None:
|
97
|
+
value_copy = copy.deepcopy(value)
|
98
|
+
template_config[attribute] = func(value=value_copy)
|
99
|
+
|
100
|
+
for key in template_config.copy():
|
101
|
+
if key not in ATTR_FUNC_MAPPING:
|
102
|
+
template_config.pop(key, None)
|
103
|
+
|
104
|
+
return template_config
|
105
|
+
|
106
|
+
def build_template_config(self) -> dict:
|
107
|
+
"""Builds the template configuration for the custom component.
|
108
|
+
|
109
|
+
Returns:
|
110
|
+
A dictionary representing the template configuration.
|
111
|
+
"""
|
112
|
+
if not self._code:
|
113
|
+
return {}
|
114
|
+
|
115
|
+
try:
|
116
|
+
cc_class = eval_custom_component_code(self._code)
|
117
|
+
|
118
|
+
except AttributeError as e:
|
119
|
+
pattern = r"module '.*?' has no attribute '.*?'"
|
120
|
+
if re.search(pattern, str(e)):
|
121
|
+
raise ImportError(e) from e
|
122
|
+
raise
|
123
|
+
|
124
|
+
component_instance = cc_class(_code=self._code)
|
125
|
+
return self.get_template_config(component_instance)
|
126
|
+
|
127
|
+
def build(self, *args: Any, **kwargs: Any) -> Any:
|
128
|
+
raise NotImplementedError
|