intentkit 0.8.16.dev1__py3-none-any.whl → 0.8.17.dev2__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 intentkit might be problematic. Click here for more details.
- intentkit/__init__.py +1 -1
- intentkit/abstracts/agent.py +4 -5
- intentkit/abstracts/engine.py +5 -5
- intentkit/abstracts/graph.py +6 -5
- intentkit/abstracts/skill.py +5 -5
- intentkit/abstracts/twitter.py +4 -5
- intentkit/clients/cdp.py +19 -77
- intentkit/clients/twitter.py +26 -34
- intentkit/clients/web3.py +1 -3
- intentkit/config/config.py +4 -0
- intentkit/core/agent.py +15 -15
- intentkit/core/asset.py +1 -2
- intentkit/core/client.py +1 -1
- intentkit/core/credit.py +19 -20
- intentkit/core/engine.py +2 -4
- intentkit/core/node.py +2 -1
- intentkit/core/prompt.py +3 -4
- intentkit/core/scheduler.py +1 -1
- intentkit/core/statistics.py +6 -7
- intentkit/models/agent.py +125 -92
- intentkit/models/agent_data.py +62 -36
- intentkit/models/app_setting.py +6 -6
- intentkit/models/chat.py +27 -24
- intentkit/models/conversation.py +8 -8
- intentkit/models/credit.py +62 -64
- intentkit/models/db.py +8 -7
- intentkit/models/db_mig.py +2 -2
- intentkit/models/llm.py +12 -14
- intentkit/models/redis.py +2 -3
- intentkit/models/skill.py +25 -27
- intentkit/models/skills.csv +29 -28
- intentkit/models/user.py +21 -22
- intentkit/skills/acolyt/ask.py +3 -4
- intentkit/skills/acolyt/base.py +1 -3
- intentkit/skills/aixbt/base.py +1 -3
- intentkit/skills/aixbt/projects.py +13 -13
- intentkit/skills/allora/base.py +1 -3
- intentkit/skills/allora/price.py +2 -3
- intentkit/skills/base.py +15 -22
- intentkit/skills/basename/__init__.py +3 -5
- intentkit/skills/carv/__init__.py +7 -8
- intentkit/skills/carv/base.py +6 -6
- intentkit/skills/carv/fetch_news.py +3 -3
- intentkit/skills/carv/onchain_query.py +4 -4
- intentkit/skills/carv/token_info_and_price.py +5 -5
- intentkit/skills/casino/base.py +1 -3
- intentkit/skills/casino/deck_draw.py +1 -2
- intentkit/skills/casino/deck_shuffle.py +1 -2
- intentkit/skills/casino/dice_roll.py +1 -2
- intentkit/skills/cdp/__init__.py +3 -5
- intentkit/skills/cdp/base.py +1 -3
- intentkit/skills/chainlist/base.py +1 -3
- intentkit/skills/chainlist/chain_lookup.py +18 -18
- intentkit/skills/common/base.py +1 -3
- intentkit/skills/common/current_time.py +1 -2
- intentkit/skills/cookiefun/base.py +1 -2
- intentkit/skills/cookiefun/get_account_details.py +7 -7
- intentkit/skills/cookiefun/get_account_feed.py +19 -19
- intentkit/skills/cookiefun/get_account_smart_followers.py +7 -7
- intentkit/skills/cookiefun/get_sectors.py +3 -3
- intentkit/skills/cookiefun/search_accounts.py +9 -9
- intentkit/skills/cryptocompare/api.py +2 -3
- intentkit/skills/cryptocompare/base.py +6 -6
- intentkit/skills/cryptocompare/fetch_news.py +3 -4
- intentkit/skills/cryptocompare/fetch_price.py +5 -6
- intentkit/skills/cryptocompare/fetch_top_exchanges.py +3 -4
- intentkit/skills/cryptocompare/fetch_top_market_cap.py +3 -4
- intentkit/skills/cryptocompare/fetch_top_volume.py +3 -4
- intentkit/skills/cryptocompare/fetch_trading_signals.py +4 -5
- intentkit/skills/cryptopanic/__init__.py +4 -4
- intentkit/skills/cryptopanic/base.py +1 -3
- intentkit/skills/cryptopanic/fetch_crypto_news.py +3 -5
- intentkit/skills/cryptopanic/fetch_crypto_sentiment.py +3 -3
- intentkit/skills/dapplooker/base.py +1 -3
- intentkit/skills/dapplooker/dapplooker_token_data.py +7 -7
- intentkit/skills/defillama/api.py +6 -9
- intentkit/skills/defillama/base.py +5 -6
- intentkit/skills/defillama/coins/fetch_batch_historical_prices.py +6 -8
- intentkit/skills/defillama/coins/fetch_block.py +4 -6
- intentkit/skills/defillama/coins/fetch_current_prices.py +6 -8
- intentkit/skills/defillama/coins/fetch_first_price.py +5 -7
- intentkit/skills/defillama/coins/fetch_historical_prices.py +7 -9
- intentkit/skills/defillama/coins/fetch_price_chart.py +7 -9
- intentkit/skills/defillama/coins/fetch_price_percentage.py +5 -7
- intentkit/skills/defillama/config/chains.py +1 -3
- intentkit/skills/defillama/fees/fetch_fees_overview.py +22 -24
- intentkit/skills/defillama/stablecoins/fetch_stablecoin_chains.py +14 -16
- intentkit/skills/defillama/stablecoins/fetch_stablecoin_charts.py +6 -8
- intentkit/skills/defillama/stablecoins/fetch_stablecoin_prices.py +3 -5
- intentkit/skills/defillama/stablecoins/fetch_stablecoins.py +5 -7
- intentkit/skills/defillama/tests/api_integration.test.py +1 -1
- intentkit/skills/defillama/tvl/fetch_chain_historical_tvl.py +2 -4
- intentkit/skills/defillama/tvl/fetch_chains.py +7 -9
- intentkit/skills/defillama/tvl/fetch_historical_tvl.py +2 -4
- intentkit/skills/defillama/tvl/fetch_protocol.py +30 -36
- intentkit/skills/defillama/tvl/fetch_protocol_current_tvl.py +1 -3
- intentkit/skills/defillama/tvl/fetch_protocols.py +35 -43
- intentkit/skills/defillama/volumes/fetch_dex_overview.py +40 -46
- intentkit/skills/defillama/volumes/fetch_dex_summary.py +33 -35
- intentkit/skills/defillama/volumes/fetch_options_overview.py +22 -26
- intentkit/skills/defillama/yields/fetch_pool_chart.py +8 -10
- intentkit/skills/defillama/yields/fetch_pools.py +24 -28
- intentkit/skills/dexscreener/__init__.py +2 -2
- intentkit/skills/dexscreener/base.py +3 -3
- intentkit/skills/dexscreener/get_pair_info.py +2 -2
- intentkit/skills/dexscreener/get_token_pairs.py +2 -2
- intentkit/skills/dexscreener/get_tokens_info.py +5 -5
- intentkit/skills/dexscreener/model/search_token_response.py +80 -82
- intentkit/skills/dexscreener/search_token.py +182 -182
- intentkit/skills/dexscreener/utils.py +15 -14
- intentkit/skills/dune_analytics/__init__.py +4 -4
- intentkit/skills/dune_analytics/base.py +1 -3
- intentkit/skills/dune_analytics/fetch_kol_buys.py +4 -4
- intentkit/skills/dune_analytics/fetch_nation_metrics.py +5 -5
- intentkit/skills/elfa/base.py +1 -3
- intentkit/skills/elfa/mention.py +19 -21
- intentkit/skills/elfa/stats.py +4 -4
- intentkit/skills/elfa/tokens.py +12 -12
- intentkit/skills/elfa/utils.py +25 -27
- intentkit/skills/enso/__init__.py +2 -2
- intentkit/skills/enso/base.py +5 -8
- intentkit/skills/enso/best_yield.py +4 -6
- intentkit/skills/enso/networks.py +1 -2
- intentkit/skills/enso/prices.py +1 -3
- intentkit/skills/enso/route.py +1 -3
- intentkit/skills/enso/tokens.py +1 -3
- intentkit/skills/enso/wallet.py +5 -5
- intentkit/skills/erc20/__init__.py +4 -6
- intentkit/skills/erc721/__init__.py +4 -6
- intentkit/skills/firecrawl/base.py +1 -3
- intentkit/skills/firecrawl/clear.py +1 -2
- intentkit/skills/firecrawl/crawl.py +9 -10
- intentkit/skills/firecrawl/query.py +1 -2
- intentkit/skills/firecrawl/scrape.py +7 -8
- intentkit/skills/firecrawl/utils.py +13 -13
- intentkit/skills/github/base.py +1 -3
- intentkit/skills/github/github_search.py +1 -2
- intentkit/skills/heurist/base.py +1 -3
- intentkit/skills/heurist/image_generation_animagine_xl.py +7 -8
- intentkit/skills/heurist/image_generation_arthemy_comics.py +7 -8
- intentkit/skills/heurist/image_generation_arthemy_real.py +7 -8
- intentkit/skills/heurist/image_generation_braindance.py +7 -8
- intentkit/skills/heurist/image_generation_cyber_realistic_xl.py +7 -8
- intentkit/skills/heurist/image_generation_flux_1_dev.py +7 -8
- intentkit/skills/heurist/image_generation_sdxl.py +7 -8
- intentkit/skills/http/base.py +1 -3
- intentkit/skills/http/get.py +7 -7
- intentkit/skills/http/post.py +9 -9
- intentkit/skills/http/put.py +9 -9
- intentkit/skills/lifi/__init__.py +4 -4
- intentkit/skills/lifi/base.py +1 -3
- intentkit/skills/lifi/token_execute.py +13 -13
- intentkit/skills/lifi/token_quote.py +6 -6
- intentkit/skills/lifi/utils.py +16 -16
- intentkit/skills/moralis/__init__.py +3 -3
- intentkit/skills/moralis/api.py +6 -7
- intentkit/skills/moralis/base.py +2 -4
- intentkit/skills/moralis/fetch_chain_portfolio.py +10 -11
- intentkit/skills/moralis/fetch_nft_portfolio.py +22 -22
- intentkit/skills/moralis/fetch_solana_portfolio.py +11 -12
- intentkit/skills/moralis/fetch_wallet_portfolio.py +8 -9
- intentkit/skills/morpho/__init__.py +4 -6
- intentkit/skills/nation/__init__.py +2 -2
- intentkit/skills/nation/base.py +1 -3
- intentkit/skills/nation/nft_check.py +3 -4
- intentkit/skills/onchain.py +2 -6
- intentkit/skills/openai/base.py +1 -3
- intentkit/skills/openai/dalle_image_generation.py +1 -3
- intentkit/skills/openai/gpt_image_generation.py +2 -3
- intentkit/skills/openai/gpt_image_to_image.py +2 -3
- intentkit/skills/openai/image_to_text.py +1 -2
- intentkit/skills/portfolio/base.py +6 -6
- intentkit/skills/portfolio/token_balances.py +21 -21
- intentkit/skills/portfolio/wallet_approvals.py +7 -7
- intentkit/skills/portfolio/wallet_defi_positions.py +3 -3
- intentkit/skills/portfolio/wallet_history.py +21 -21
- intentkit/skills/portfolio/wallet_net_worth.py +13 -13
- intentkit/skills/portfolio/wallet_nfts.py +19 -19
- intentkit/skills/portfolio/wallet_profitability.py +7 -7
- intentkit/skills/portfolio/wallet_profitability_summary.py +5 -5
- intentkit/skills/portfolio/wallet_stats.py +3 -3
- intentkit/skills/portfolio/wallet_swaps.py +19 -19
- intentkit/skills/pyth/__init__.py +3 -5
- intentkit/skills/slack/base.py +2 -4
- intentkit/skills/slack/get_channel.py +8 -8
- intentkit/skills/slack/get_message.py +9 -9
- intentkit/skills/slack/schedule_message.py +5 -5
- intentkit/skills/slack/send_message.py +3 -5
- intentkit/skills/supabase/base.py +1 -3
- intentkit/skills/supabase/delete_data.py +4 -4
- intentkit/skills/supabase/fetch_data.py +12 -12
- intentkit/skills/supabase/insert_data.py +4 -4
- intentkit/skills/supabase/invoke_function.py +6 -6
- intentkit/skills/supabase/update_data.py +6 -6
- intentkit/skills/supabase/upsert_data.py +4 -4
- intentkit/skills/superfluid/__init__.py +4 -6
- intentkit/skills/system/add_autonomous_task.py +8 -10
- intentkit/skills/system/edit_autonomous_task.py +12 -14
- intentkit/skills/system/list_autonomous_tasks.py +1 -3
- intentkit/skills/tavily/base.py +1 -3
- intentkit/skills/tavily/tavily_extract.py +1 -2
- intentkit/skills/tavily/tavily_search.py +1 -3
- intentkit/skills/token/base.py +5 -5
- intentkit/skills/token/erc20_transfers.py +19 -19
- intentkit/skills/token/token_analytics.py +3 -3
- intentkit/skills/token/token_price.py +13 -13
- intentkit/skills/token/token_search.py +9 -9
- intentkit/skills/twitter/base.py +3 -4
- intentkit/skills/twitter/follow_user.py +1 -2
- intentkit/skills/twitter/get_mentions.py +3 -4
- intentkit/skills/twitter/get_timeline.py +1 -2
- intentkit/skills/twitter/get_user_by_username.py +1 -2
- intentkit/skills/twitter/get_user_tweets.py +2 -3
- intentkit/skills/twitter/like_tweet.py +1 -2
- intentkit/skills/twitter/post_tweet.py +3 -4
- intentkit/skills/twitter/reply_tweet.py +3 -4
- intentkit/skills/twitter/retweet.py +1 -2
- intentkit/skills/twitter/search_tweets.py +1 -2
- intentkit/skills/unrealspeech/base.py +1 -3
- intentkit/skills/unrealspeech/text_to_speech.py +8 -8
- intentkit/skills/venice_audio/__init__.py +8 -9
- intentkit/skills/venice_audio/base.py +3 -4
- intentkit/skills/venice_audio/input.py +41 -41
- intentkit/skills/venice_audio/venice_audio.py +6 -6
- intentkit/skills/venice_image/__init__.py +5 -5
- intentkit/skills/venice_image/api.py +138 -138
- intentkit/skills/venice_image/base.py +3 -3
- intentkit/skills/venice_image/config.py +33 -35
- intentkit/skills/venice_image/image_enhance/image_enhance.py +2 -3
- intentkit/skills/venice_image/image_enhance/image_enhance_base.py +21 -23
- intentkit/skills/venice_image/image_enhance/image_enhance_input.py +38 -40
- intentkit/skills/venice_image/image_generation/image_generation_base.py +9 -9
- intentkit/skills/venice_image/image_generation/image_generation_fluently_xl.py +26 -26
- intentkit/skills/venice_image/image_generation/image_generation_flux_dev.py +27 -27
- intentkit/skills/venice_image/image_generation/image_generation_flux_dev_uncensored.py +26 -26
- intentkit/skills/venice_image/image_generation/image_generation_input.py +158 -158
- intentkit/skills/venice_image/image_generation/image_generation_lustify_sdxl.py +26 -26
- intentkit/skills/venice_image/image_generation/image_generation_pony_realism.py +26 -26
- intentkit/skills/venice_image/image_generation/image_generation_stable_diffusion_3_5.py +28 -28
- intentkit/skills/venice_image/image_generation/image_generation_venice_sd35.py +28 -28
- intentkit/skills/venice_image/image_upscale/image_upscale.py +3 -3
- intentkit/skills/venice_image/image_upscale/image_upscale_base.py +21 -23
- intentkit/skills/venice_image/image_upscale/image_upscale_input.py +22 -22
- intentkit/skills/venice_image/image_vision/image_vision.py +2 -2
- intentkit/skills/venice_image/image_vision/image_vision_base.py +17 -17
- intentkit/skills/venice_image/image_vision/image_vision_input.py +9 -9
- intentkit/skills/venice_image/utils.py +77 -78
- intentkit/skills/web_scraper/base.py +1 -3
- intentkit/skills/web_scraper/document_indexer.py +1 -2
- intentkit/skills/web_scraper/scrape_and_index.py +4 -5
- intentkit/skills/web_scraper/utils.py +25 -26
- intentkit/skills/web_scraper/website_indexer.py +10 -11
- intentkit/skills/weth/__init__.py +4 -6
- intentkit/skills/wow/__init__.py +4 -6
- intentkit/skills/x402/__init__.py +11 -3
- intentkit/skills/x402/ask_agent.py +12 -78
- intentkit/skills/x402/base.py +90 -0
- intentkit/skills/x402/http_request.py +117 -0
- intentkit/skills/x402/schema.json +15 -10
- intentkit/skills/xmtp/base.py +3 -3
- intentkit/skills/xmtp/price.py +2 -2
- intentkit/skills/xmtp/swap.py +2 -4
- intentkit/skills/xmtp/transfer.py +4 -6
- intentkit/utils/error.py +2 -2
- intentkit/utils/logging.py +2 -4
- intentkit/utils/s3.py +8 -9
- intentkit/utils/schema.py +5 -5
- intentkit/utils/slack_alert.py +7 -8
- {intentkit-0.8.16.dev1.dist-info → intentkit-0.8.17.dev2.dist-info}/METADATA +3 -4
- intentkit-0.8.17.dev2.dist-info/RECORD +464 -0
- intentkit/models/generator.py +0 -347
- intentkit-0.8.16.dev1.dist-info/RECORD +0 -464
- {intentkit-0.8.16.dev1.dist-info → intentkit-0.8.17.dev2.dist-info}/WHEEL +0 -0
- {intentkit-0.8.16.dev1.dist-info → intentkit-0.8.17.dev2.dist-info}/licenses/LICENSE +0 -0
|
@@ -1,138 +1,138 @@
|
|
|
1
|
-
"""
|
|
2
|
-
This module encapsulates all interactions with the Venice AI API.
|
|
3
|
-
It provides a function, make_venice_api_request, to make POST requests
|
|
4
|
-
to the API and handles the responses, including error handling,
|
|
5
|
-
content type checking, and image storage via S3. This separation
|
|
6
|
-
of concerns keeps the core skill logic cleaner and easier to maintain.
|
|
7
|
-
"""
|
|
8
|
-
|
|
9
|
-
import hashlib
|
|
10
|
-
import logging
|
|
11
|
-
from typing import Any
|
|
12
|
-
|
|
13
|
-
import httpx
|
|
14
|
-
|
|
15
|
-
from intentkit.utils.s3 import store_image_bytes
|
|
16
|
-
|
|
17
|
-
logger = logging.getLogger(__name__)
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
async def make_venice_api_request(
|
|
21
|
-
api_key: str,
|
|
22
|
-
path: str,
|
|
23
|
-
payload:
|
|
24
|
-
category: str,
|
|
25
|
-
tool_name: str,
|
|
26
|
-
) ->
|
|
27
|
-
"""
|
|
28
|
-
Makes a POST request to the Venice AI API, handling all aspects
|
|
29
|
-
of the API interaction. This includes:
|
|
30
|
-
|
|
31
|
-
1. Constructing the API URL using a base URL and the provided path.
|
|
32
|
-
2. Adding the required authorization header with the provided API key.
|
|
33
|
-
3. Sending the POST request with the given payload.
|
|
34
|
-
4. Handling potential connection and HTTP errors.
|
|
35
|
-
5. Calling the internal _handle_response function to process the API's
|
|
36
|
-
response, which might be JSON or an image.
|
|
37
|
-
|
|
38
|
-
Args:
|
|
39
|
-
api_key: The Venice AI API key for authentication.
|
|
40
|
-
path: The API endpoint path (e.g., "/api/v1/image/generate"). Should *not* start with the base URL.
|
|
41
|
-
payload: The data to send in the request body (as JSON).
|
|
42
|
-
category: The category of the skill making the request (e.g., "venice_image"). Used for S3 storage and logging purpose.
|
|
43
|
-
tool_name: The name of the tool or skill making the request (e.g., "image_generation"). Used for S3 storage and logging purpose.
|
|
44
|
-
|
|
45
|
-
Returns:
|
|
46
|
-
A tuple: (success_data, error_data).
|
|
47
|
-
- success_data: A dictionary containing the parsed JSON response from the API if the request was successful
|
|
48
|
-
(or a dictionary containing the S3 URL if the response is an image).
|
|
49
|
-
- error_data: A dictionary containing information about any errors that occurred,
|
|
50
|
-
or None if the request was successful. The dictionary includes an 'error' key.
|
|
51
|
-
"""
|
|
52
|
-
|
|
53
|
-
venice_base_url = "https://api.venice.ai" # Venice AI API base URL
|
|
54
|
-
|
|
55
|
-
if not path.startswith("/"):
|
|
56
|
-
path = "/" + path
|
|
57
|
-
|
|
58
|
-
api_url = f"{venice_base_url}{path}"
|
|
59
|
-
|
|
60
|
-
headers = {
|
|
61
|
-
"Authorization": f"Bearer {api_key}",
|
|
62
|
-
"Content-Type": "application/json",
|
|
63
|
-
"Accept": "image/*, application/json",
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
logger.info(
|
|
67
|
-
f"[{category}/{tool_name}] Sending request to {api_url} with payload: {payload}"
|
|
68
|
-
)
|
|
69
|
-
|
|
70
|
-
try:
|
|
71
|
-
async with httpx.AsyncClient(timeout=180.0) as client:
|
|
72
|
-
response = await client.post(api_url, json=payload, headers=headers)
|
|
73
|
-
return await _handle_response(response, category, tool_name)
|
|
74
|
-
|
|
75
|
-
except httpx.RequestError as e:
|
|
76
|
-
error_msg = f"Connection error: {e}"
|
|
77
|
-
logger.error(f"[{category}/{tool_name}] {error_msg}")
|
|
78
|
-
return {}, {"success": False, "error": error_msg}
|
|
79
|
-
except Exception as e:
|
|
80
|
-
error_msg = f"Unexpected error: {e}"
|
|
81
|
-
logger.error(f"[{category}/{tool_name}] {error_msg}", exc_info=True)
|
|
82
|
-
return {}, {"success": False, "error": error_msg}
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
async def _handle_response(
|
|
86
|
-
response: httpx.Response, category: str, tool_name: str
|
|
87
|
-
) ->
|
|
88
|
-
"""
|
|
89
|
-
Handles the API response, differentiating between JSON and image responses.
|
|
90
|
-
|
|
91
|
-
If the response is an image (based on the 'content-type' header),
|
|
92
|
-
it stores the image in S3 and returns the S3 URL.
|
|
93
|
-
If the response is JSON, it parses the JSON and returns it.
|
|
94
|
-
If any errors occur, it returns an error dictionary.
|
|
95
|
-
"""
|
|
96
|
-
|
|
97
|
-
content_type = str(response.headers.get("content-type", "")).lower()
|
|
98
|
-
|
|
99
|
-
if response.status_code == 200 and content_type.startswith("image/"):
|
|
100
|
-
try:
|
|
101
|
-
upscaled_image_bytes = response.content
|
|
102
|
-
image_hash = hashlib.sha256(upscaled_image_bytes).hexdigest()
|
|
103
|
-
file_extension = content_type.split("/")[-1].split("+")[0] or "png"
|
|
104
|
-
|
|
105
|
-
key = f"{category}/{tool_name}/{image_hash}.{file_extension}"
|
|
106
|
-
|
|
107
|
-
logger.info(f"[{category}/{tool_name}] Storing image with key: {key}")
|
|
108
|
-
|
|
109
|
-
stored_url = await store_image_bytes(
|
|
110
|
-
upscaled_image_bytes, key, content_type=content_type
|
|
111
|
-
)
|
|
112
|
-
|
|
113
|
-
return {"success": True, "result": stored_url}, None
|
|
114
|
-
|
|
115
|
-
except Exception as e:
|
|
116
|
-
error_msg = f"Error processing image response: {e}"
|
|
117
|
-
logger.error(f"[{category}/{tool_name}] {error_msg}", exc_info=True)
|
|
118
|
-
return {}, {"success": False, "error": error_msg}
|
|
119
|
-
|
|
120
|
-
elif response.status_code == 200:
|
|
121
|
-
try:
|
|
122
|
-
logger.info(f"[{category}/{tool_name}] Received successful JSON response.")
|
|
123
|
-
return response.json(), None
|
|
124
|
-
except Exception as json_err:
|
|
125
|
-
error_msg = f"Failed to parse JSON response: {json_err} - {response.text}"
|
|
126
|
-
logger.error(f"[{category}/{tool_name}] {error_msg}")
|
|
127
|
-
return {}, {"success": False, "error": error_msg}
|
|
128
|
-
|
|
129
|
-
else:
|
|
130
|
-
try:
|
|
131
|
-
error_data = response.json()
|
|
132
|
-
error_msg = f"API returned error: {error_data.get('message', error_data.get('detail', response.text))}"
|
|
133
|
-
logger.error(f"[{category}/{tool_name}] {error_msg}")
|
|
134
|
-
return {}, {"success": False, "error": error_msg}
|
|
135
|
-
except Exception:
|
|
136
|
-
error_msg = f"API returned status code {response.status_code} with text: {response.text}"
|
|
137
|
-
logger.error(f"[{category}/{tool_name}] {error_msg}")
|
|
138
|
-
return {}, {"success": False, "error": error_msg}
|
|
1
|
+
"""
|
|
2
|
+
This module encapsulates all interactions with the Venice AI API.
|
|
3
|
+
It provides a function, make_venice_api_request, to make POST requests
|
|
4
|
+
to the API and handles the responses, including error handling,
|
|
5
|
+
content type checking, and image storage via S3. This separation
|
|
6
|
+
of concerns keeps the core skill logic cleaner and easier to maintain.
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
import hashlib
|
|
10
|
+
import logging
|
|
11
|
+
from typing import Any
|
|
12
|
+
|
|
13
|
+
import httpx
|
|
14
|
+
|
|
15
|
+
from intentkit.utils.s3 import store_image_bytes
|
|
16
|
+
|
|
17
|
+
logger = logging.getLogger(__name__)
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
async def make_venice_api_request(
|
|
21
|
+
api_key: str,
|
|
22
|
+
path: str,
|
|
23
|
+
payload: dict[str, Any],
|
|
24
|
+
category: str,
|
|
25
|
+
tool_name: str,
|
|
26
|
+
) -> tuple[dict[str, Any], dict[str, Any] | None]:
|
|
27
|
+
"""
|
|
28
|
+
Makes a POST request to the Venice AI API, handling all aspects
|
|
29
|
+
of the API interaction. This includes:
|
|
30
|
+
|
|
31
|
+
1. Constructing the API URL using a base URL and the provided path.
|
|
32
|
+
2. Adding the required authorization header with the provided API key.
|
|
33
|
+
3. Sending the POST request with the given payload.
|
|
34
|
+
4. Handling potential connection and HTTP errors.
|
|
35
|
+
5. Calling the internal _handle_response function to process the API's
|
|
36
|
+
response, which might be JSON or an image.
|
|
37
|
+
|
|
38
|
+
Args:
|
|
39
|
+
api_key: The Venice AI API key for authentication.
|
|
40
|
+
path: The API endpoint path (e.g., "/api/v1/image/generate"). Should *not* start with the base URL.
|
|
41
|
+
payload: The data to send in the request body (as JSON).
|
|
42
|
+
category: The category of the skill making the request (e.g., "venice_image"). Used for S3 storage and logging purpose.
|
|
43
|
+
tool_name: The name of the tool or skill making the request (e.g., "image_generation"). Used for S3 storage and logging purpose.
|
|
44
|
+
|
|
45
|
+
Returns:
|
|
46
|
+
A tuple: (success_data, error_data).
|
|
47
|
+
- success_data: A dictionary containing the parsed JSON response from the API if the request was successful
|
|
48
|
+
(or a dictionary containing the S3 URL if the response is an image).
|
|
49
|
+
- error_data: A dictionary containing information about any errors that occurred,
|
|
50
|
+
or None if the request was successful. The dictionary includes an 'error' key.
|
|
51
|
+
"""
|
|
52
|
+
|
|
53
|
+
venice_base_url = "https://api.venice.ai" # Venice AI API base URL
|
|
54
|
+
|
|
55
|
+
if not path.startswith("/"):
|
|
56
|
+
path = "/" + path
|
|
57
|
+
|
|
58
|
+
api_url = f"{venice_base_url}{path}"
|
|
59
|
+
|
|
60
|
+
headers = {
|
|
61
|
+
"Authorization": f"Bearer {api_key}",
|
|
62
|
+
"Content-Type": "application/json",
|
|
63
|
+
"Accept": "image/*, application/json",
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
logger.info(
|
|
67
|
+
f"[{category}/{tool_name}] Sending request to {api_url} with payload: {payload}"
|
|
68
|
+
)
|
|
69
|
+
|
|
70
|
+
try:
|
|
71
|
+
async with httpx.AsyncClient(timeout=180.0) as client:
|
|
72
|
+
response = await client.post(api_url, json=payload, headers=headers)
|
|
73
|
+
return await _handle_response(response, category, tool_name)
|
|
74
|
+
|
|
75
|
+
except httpx.RequestError as e:
|
|
76
|
+
error_msg = f"Connection error: {e}"
|
|
77
|
+
logger.error(f"[{category}/{tool_name}] {error_msg}")
|
|
78
|
+
return {}, {"success": False, "error": error_msg}
|
|
79
|
+
except Exception as e:
|
|
80
|
+
error_msg = f"Unexpected error: {e}"
|
|
81
|
+
logger.error(f"[{category}/{tool_name}] {error_msg}", exc_info=True)
|
|
82
|
+
return {}, {"success": False, "error": error_msg}
|
|
83
|
+
|
|
84
|
+
|
|
85
|
+
async def _handle_response(
|
|
86
|
+
response: httpx.Response, category: str, tool_name: str
|
|
87
|
+
) -> tuple[dict[str, Any], dict[str, Any] | None]:
|
|
88
|
+
"""
|
|
89
|
+
Handles the API response, differentiating between JSON and image responses.
|
|
90
|
+
|
|
91
|
+
If the response is an image (based on the 'content-type' header),
|
|
92
|
+
it stores the image in S3 and returns the S3 URL.
|
|
93
|
+
If the response is JSON, it parses the JSON and returns it.
|
|
94
|
+
If any errors occur, it returns an error dictionary.
|
|
95
|
+
"""
|
|
96
|
+
|
|
97
|
+
content_type = str(response.headers.get("content-type", "")).lower()
|
|
98
|
+
|
|
99
|
+
if response.status_code == 200 and content_type.startswith("image/"):
|
|
100
|
+
try:
|
|
101
|
+
upscaled_image_bytes = response.content
|
|
102
|
+
image_hash = hashlib.sha256(upscaled_image_bytes).hexdigest()
|
|
103
|
+
file_extension = content_type.split("/")[-1].split("+")[0] or "png"
|
|
104
|
+
|
|
105
|
+
key = f"{category}/{tool_name}/{image_hash}.{file_extension}"
|
|
106
|
+
|
|
107
|
+
logger.info(f"[{category}/{tool_name}] Storing image with key: {key}")
|
|
108
|
+
|
|
109
|
+
stored_url = await store_image_bytes(
|
|
110
|
+
upscaled_image_bytes, key, content_type=content_type
|
|
111
|
+
)
|
|
112
|
+
|
|
113
|
+
return {"success": True, "result": stored_url}, None
|
|
114
|
+
|
|
115
|
+
except Exception as e:
|
|
116
|
+
error_msg = f"Error processing image response: {e}"
|
|
117
|
+
logger.error(f"[{category}/{tool_name}] {error_msg}", exc_info=True)
|
|
118
|
+
return {}, {"success": False, "error": error_msg}
|
|
119
|
+
|
|
120
|
+
elif response.status_code == 200:
|
|
121
|
+
try:
|
|
122
|
+
logger.info(f"[{category}/{tool_name}] Received successful JSON response.")
|
|
123
|
+
return response.json(), None
|
|
124
|
+
except Exception as json_err:
|
|
125
|
+
error_msg = f"Failed to parse JSON response: {json_err} - {response.text}"
|
|
126
|
+
logger.error(f"[{category}/{tool_name}] {error_msg}")
|
|
127
|
+
return {}, {"success": False, "error": error_msg}
|
|
128
|
+
|
|
129
|
+
else:
|
|
130
|
+
try:
|
|
131
|
+
error_data = response.json()
|
|
132
|
+
error_msg = f"API returned error: {error_data.get('message', error_data.get('detail', response.text))}"
|
|
133
|
+
logger.error(f"[{category}/{tool_name}] {error_msg}")
|
|
134
|
+
return {}, {"success": False, "error": error_msg}
|
|
135
|
+
except Exception:
|
|
136
|
+
error_msg = f"API returned status code {response.status_code} with text: {response.text}"
|
|
137
|
+
logger.error(f"[{category}/{tool_name}] {error_msg}")
|
|
138
|
+
return {}, {"success": False, "error": error_msg}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import logging
|
|
2
|
-
from typing import Any
|
|
2
|
+
from typing import Any
|
|
3
3
|
|
|
4
4
|
from langchain_core.tools.base import ToolException
|
|
5
5
|
|
|
@@ -155,8 +155,8 @@ class VeniceImageBaseTool(IntentKitSkill):
|
|
|
155
155
|
raise ToolException(f"Failed to apply Venice rate limit: {str(e)}") from e
|
|
156
156
|
|
|
157
157
|
async def post(
|
|
158
|
-
self, path: str, payload:
|
|
159
|
-
) ->
|
|
158
|
+
self, path: str, payload: dict[str, Any], context
|
|
159
|
+
) -> tuple[dict[str, Any], dict[str, Any] | None]:
|
|
160
160
|
"""
|
|
161
161
|
Makes a POST request to the Venice AI API using the `make_venice_api_request`
|
|
162
162
|
function from the `skills.venice_image.api` module.
|
|
@@ -1,35 +1,33 @@
|
|
|
1
|
-
from
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
description="Duration of the time window (in minutes) for rate limiting.",
|
|
35
|
-
)
|
|
1
|
+
from pydantic import BaseModel, Field
|
|
2
|
+
|
|
3
|
+
|
|
4
|
+
class VeniceImageConfig(BaseModel):
|
|
5
|
+
"""Skill Config for Venice Image."""
|
|
6
|
+
|
|
7
|
+
api_key_provider: str = Field(
|
|
8
|
+
default="agent_owner",
|
|
9
|
+
description="Provider of the API Key, could be agent_owner or platform",
|
|
10
|
+
)
|
|
11
|
+
safe_mode: bool = Field(
|
|
12
|
+
default=True,
|
|
13
|
+
description="Whether to use safe mode. If enabled, this will blur images that are classified as having adult content",
|
|
14
|
+
)
|
|
15
|
+
hide_watermark: bool = Field(
|
|
16
|
+
default=True,
|
|
17
|
+
description="Whether to hide the Venice watermark. Venice may ignore this parameter for certain generated content.",
|
|
18
|
+
)
|
|
19
|
+
embed_exif_metadata: bool = Field(
|
|
20
|
+
default=False, description="Whether to embed EXIF metadata in the image."
|
|
21
|
+
)
|
|
22
|
+
negative_prompt: str = Field(
|
|
23
|
+
default="(worst quality: 1.4), bad quality, nsfw",
|
|
24
|
+
description="The default negative prompt used when no other prompt is provided.",
|
|
25
|
+
)
|
|
26
|
+
rate_limit_number: int | None = Field(
|
|
27
|
+
default=None,
|
|
28
|
+
description="Maximum number of allowed calls within the specified time window.",
|
|
29
|
+
)
|
|
30
|
+
rate_limit_minutes: int | None = Field(
|
|
31
|
+
default=None,
|
|
32
|
+
description="Duration of the time window (in minutes) for rate limiting.",
|
|
33
|
+
)
|
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
import logging
|
|
2
|
-
from typing import Optional
|
|
3
2
|
|
|
4
3
|
from pydantic import HttpUrl
|
|
5
4
|
|
|
@@ -30,8 +29,8 @@ class ImageEnhance(VeniceImageEnhanceBaseTool):
|
|
|
30
29
|
self,
|
|
31
30
|
image_url: HttpUrl,
|
|
32
31
|
enhancePrompt: str,
|
|
33
|
-
replication:
|
|
34
|
-
enhanceCreativity:
|
|
32
|
+
replication: float | None = 0.35,
|
|
33
|
+
enhanceCreativity: float | None = 0.5,
|
|
35
34
|
**kwargs,
|
|
36
35
|
) -> dict:
|
|
37
36
|
"""
|
|
@@ -1,23 +1,21 @@
|
|
|
1
|
-
from
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
class
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
description="A description of what the image Enchanching tool does."
|
|
23
|
-
)
|
|
1
|
+
from pydantic import BaseModel, Field
|
|
2
|
+
|
|
3
|
+
# Import the generic base and shared input
|
|
4
|
+
from intentkit.skills.venice_image.base import VeniceImageBaseTool
|
|
5
|
+
from intentkit.skills.venice_image.image_enhance.image_enhance_input import (
|
|
6
|
+
VeniceImageEnhanceInput,
|
|
7
|
+
)
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
class VeniceImageEnhanceBaseTool(VeniceImageBaseTool):
|
|
11
|
+
"""
|
|
12
|
+
Base class for Venice AI *Image Enchanching* tools.
|
|
13
|
+
Inherits from VeniceAIBaseTool and handles specifics of the
|
|
14
|
+
/image/upscale endpoint
|
|
15
|
+
"""
|
|
16
|
+
|
|
17
|
+
args_schema: type[BaseModel] = VeniceImageEnhanceInput
|
|
18
|
+
name: str = Field(description="The unique name of the image Enchanching tool.")
|
|
19
|
+
description: str = Field(
|
|
20
|
+
description="A description of what the image Enchanching tool does."
|
|
21
|
+
)
|
|
@@ -1,40 +1,38 @@
|
|
|
1
|
-
from
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
),
|
|
40
|
-
)
|
|
1
|
+
from pydantic import BaseModel, Field
|
|
2
|
+
|
|
3
|
+
|
|
4
|
+
class VeniceImageEnhanceInput(BaseModel):
|
|
5
|
+
"""Input for Venice Image Enhance tool (scale=1, enhance=True)."""
|
|
6
|
+
|
|
7
|
+
image_url: str = Field(
|
|
8
|
+
description="The URL of the image to enhance. Must be a publicly accessible URL."
|
|
9
|
+
)
|
|
10
|
+
|
|
11
|
+
enhancePrompt: str = Field(
|
|
12
|
+
...,
|
|
13
|
+
max_length=1500,
|
|
14
|
+
description=(
|
|
15
|
+
"Required prompt describing the desired enhancement style. "
|
|
16
|
+
"Best used with short descriptors like 'gold', 'marble', or 'angry, menacing'."
|
|
17
|
+
),
|
|
18
|
+
)
|
|
19
|
+
|
|
20
|
+
replication: float | None = Field(
|
|
21
|
+
default=0.35,
|
|
22
|
+
ge=0.1,
|
|
23
|
+
le=1.0,
|
|
24
|
+
description=(
|
|
25
|
+
"How strongly lines and noise in the base image are preserved. "
|
|
26
|
+
"Higher values retain more noise and detail but are less smooth."
|
|
27
|
+
),
|
|
28
|
+
)
|
|
29
|
+
|
|
30
|
+
enhanceCreativity: float = Field(
|
|
31
|
+
default=0.5,
|
|
32
|
+
ge=0.0,
|
|
33
|
+
le=1.0,
|
|
34
|
+
description=(
|
|
35
|
+
"How much the enhancement AI is allowed to change the image. "
|
|
36
|
+
"0 = minimal change, 1 = generate a new image entirely."
|
|
37
|
+
),
|
|
38
|
+
)
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import base64
|
|
2
2
|
import hashlib
|
|
3
3
|
import logging
|
|
4
|
-
from typing import Any,
|
|
4
|
+
from typing import Any, Literal
|
|
5
5
|
|
|
6
6
|
from pydantic import BaseModel, Field
|
|
7
7
|
|
|
@@ -24,7 +24,7 @@ class VeniceImageGenerationBaseTool(VeniceImageBaseTool):
|
|
|
24
24
|
"""
|
|
25
25
|
|
|
26
26
|
# --- Attributes specific to Image Generation ---
|
|
27
|
-
args_schema:
|
|
27
|
+
args_schema: type[BaseModel] = VeniceImageGenerationInput
|
|
28
28
|
|
|
29
29
|
# --- Attributes Subclasses MUST Define ---
|
|
30
30
|
name: str = Field(description="The unique name of the image generation tool/model.")
|
|
@@ -38,15 +38,15 @@ class VeniceImageGenerationBaseTool(VeniceImageBaseTool):
|
|
|
38
38
|
async def _arun(
|
|
39
39
|
self,
|
|
40
40
|
prompt: str,
|
|
41
|
-
seed:
|
|
42
|
-
negative_prompt:
|
|
43
|
-
width:
|
|
44
|
-
height:
|
|
41
|
+
seed: int | None = None,
|
|
42
|
+
negative_prompt: str | None = None,
|
|
43
|
+
width: int | None = 1024,
|
|
44
|
+
height: int | None = 1024,
|
|
45
45
|
format: Literal["png", "jpeg", "webp"] = "png",
|
|
46
|
-
cfg_scale:
|
|
47
|
-
style_preset:
|
|
46
|
+
cfg_scale: float | None = 7.5,
|
|
47
|
+
style_preset: str | None = "Photographic",
|
|
48
48
|
**kwargs,
|
|
49
|
-
) ->
|
|
49
|
+
) -> dict[str, Any]:
|
|
50
50
|
try:
|
|
51
51
|
context = self.get_context()
|
|
52
52
|
skillConfig = self.getSkillConfig(context)
|
|
@@ -1,26 +1,26 @@
|
|
|
1
|
-
from intentkit.skills.venice_image.image_generation.image_generation_base import (
|
|
2
|
-
VeniceImageGenerationBaseTool,
|
|
3
|
-
)
|
|
4
|
-
from intentkit.skills.venice_image.image_generation.image_generation_input import (
|
|
5
|
-
STYLE_PRESETS,
|
|
6
|
-
)
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
class ImageGenerationFluentlyXL(VeniceImageGenerationBaseTool):
|
|
10
|
-
"""
|
|
11
|
-
Tool for generating images using the Fluently-XL model via Venice AI.
|
|
12
|
-
Known for aesthetics, lighting, realism, and correct anatomy.
|
|
13
|
-
"""
|
|
14
|
-
|
|
15
|
-
# --- Model Specific Configuration ---
|
|
16
|
-
name: str = "venice_image_generation_fluently_xl"
|
|
17
|
-
description: str = (
|
|
18
|
-
"Generate images using the Fluently-XL model (via Venice AI).\n"
|
|
19
|
-
"Aims for improved aesthetics, lighting, realism, and anatomy. Good for professional-quality images.\n"
|
|
20
|
-
"Provide a text prompt describing the image (up to 1500 chars).\n"
|
|
21
|
-
f"Optionally specify a style preset from the list: {', '.join(STYLE_PRESETS)}.\n"
|
|
22
|
-
"Supports dimensions up to 2048x2048 (multiple of 8)."
|
|
23
|
-
)
|
|
24
|
-
model_id: str = "fluently-xl"
|
|
25
|
-
|
|
26
|
-
# args_schema and _arun are inherited from VeniceImageGenerationBaseTool
|
|
1
|
+
from intentkit.skills.venice_image.image_generation.image_generation_base import (
|
|
2
|
+
VeniceImageGenerationBaseTool,
|
|
3
|
+
)
|
|
4
|
+
from intentkit.skills.venice_image.image_generation.image_generation_input import (
|
|
5
|
+
STYLE_PRESETS,
|
|
6
|
+
)
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
class ImageGenerationFluentlyXL(VeniceImageGenerationBaseTool):
|
|
10
|
+
"""
|
|
11
|
+
Tool for generating images using the Fluently-XL model via Venice AI.
|
|
12
|
+
Known for aesthetics, lighting, realism, and correct anatomy.
|
|
13
|
+
"""
|
|
14
|
+
|
|
15
|
+
# --- Model Specific Configuration ---
|
|
16
|
+
name: str = "venice_image_generation_fluently_xl"
|
|
17
|
+
description: str = (
|
|
18
|
+
"Generate images using the Fluently-XL model (via Venice AI).\n"
|
|
19
|
+
"Aims for improved aesthetics, lighting, realism, and anatomy. Good for professional-quality images.\n"
|
|
20
|
+
"Provide a text prompt describing the image (up to 1500 chars).\n"
|
|
21
|
+
f"Optionally specify a style preset from the list: {', '.join(STYLE_PRESETS)}.\n"
|
|
22
|
+
"Supports dimensions up to 2048x2048 (multiple of 8)."
|
|
23
|
+
)
|
|
24
|
+
model_id: str = "fluently-xl"
|
|
25
|
+
|
|
26
|
+
# args_schema and _arun are inherited from VeniceImageGenerationBaseTool
|