intentkit 0.6.13.dev2__py3-none-any.whl → 0.8.17__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 +14 -7
- intentkit/abstracts/skill.py +6 -144
- intentkit/abstracts/twitter.py +4 -5
- intentkit/clients/__init__.py +5 -2
- intentkit/clients/cdp.py +101 -141
- intentkit/clients/twitter.py +83 -62
- intentkit/clients/web3.py +29 -0
- intentkit/config/config.py +8 -5
- intentkit/core/agent.py +472 -195
- intentkit/core/asset.py +253 -0
- intentkit/core/chat.py +51 -0
- intentkit/core/client.py +1 -1
- intentkit/core/credit.py +460 -130
- intentkit/core/engine.py +262 -233
- intentkit/core/node.py +15 -16
- intentkit/core/prompt.py +62 -28
- intentkit/core/scheduler.py +92 -0
- intentkit/core/statistics.py +168 -0
- intentkit/models/agent.py +1096 -949
- intentkit/models/agent_data.py +68 -38
- intentkit/models/agent_public.json +98 -0
- intentkit/models/agent_schema.json +54 -439
- intentkit/models/app_setting.py +96 -33
- intentkit/models/chat.py +74 -27
- intentkit/models/conversation.py +8 -8
- intentkit/models/credit.py +362 -74
- intentkit/models/db.py +26 -8
- intentkit/models/db_mig.py +2 -2
- intentkit/models/llm.csv +28 -0
- intentkit/models/llm.py +185 -350
- intentkit/models/redis.py +6 -4
- intentkit/models/skill.py +186 -72
- intentkit/models/skills.csv +174 -0
- intentkit/models/user.py +82 -24
- intentkit/skills/acolyt/__init__.py +2 -9
- intentkit/skills/acolyt/ask.py +3 -4
- intentkit/skills/acolyt/base.py +4 -9
- intentkit/skills/acolyt/schema.json +4 -3
- intentkit/skills/aixbt/__init__.py +2 -13
- intentkit/skills/aixbt/base.py +1 -7
- intentkit/skills/aixbt/projects.py +14 -15
- intentkit/skills/aixbt/schema.json +4 -4
- intentkit/skills/allora/__init__.py +2 -9
- intentkit/skills/allora/base.py +4 -9
- intentkit/skills/allora/price.py +3 -4
- intentkit/skills/allora/schema.json +3 -2
- intentkit/skills/base.py +248 -85
- intentkit/skills/basename/__init__.py +51 -0
- intentkit/skills/basename/base.py +11 -0
- intentkit/skills/basename/basename.svg +11 -0
- intentkit/skills/basename/schema.json +58 -0
- intentkit/skills/carv/__init__.py +115 -121
- intentkit/skills/carv/base.py +184 -185
- intentkit/skills/carv/fetch_news.py +3 -3
- intentkit/skills/carv/onchain_query.py +4 -4
- intentkit/skills/carv/schema.json +134 -137
- intentkit/skills/carv/token_info_and_price.py +5 -5
- intentkit/skills/casino/README.md +254 -0
- intentkit/skills/casino/__init__.py +86 -0
- intentkit/skills/casino/base.py +17 -0
- intentkit/skills/casino/casino.png +0 -0
- intentkit/skills/casino/deck_draw.py +127 -0
- intentkit/skills/casino/deck_shuffle.py +118 -0
- intentkit/skills/casino/dice_roll.py +100 -0
- intentkit/skills/casino/schema.json +77 -0
- intentkit/skills/casino/utils.py +107 -0
- intentkit/skills/cdp/__init__.py +22 -84
- intentkit/skills/cdp/base.py +1 -7
- intentkit/skills/cdp/schema.json +11 -314
- intentkit/skills/chainlist/__init__.py +2 -7
- intentkit/skills/chainlist/base.py +1 -7
- intentkit/skills/chainlist/chain_lookup.py +18 -18
- intentkit/skills/chainlist/schema.json +3 -5
- intentkit/skills/common/__init__.py +2 -9
- intentkit/skills/common/base.py +1 -7
- intentkit/skills/common/current_time.py +1 -2
- intentkit/skills/common/schema.json +2 -2
- intentkit/skills/cookiefun/__init__.py +6 -9
- intentkit/skills/cookiefun/base.py +2 -7
- 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/schema.json +1 -3
- intentkit/skills/cookiefun/search_accounts.py +9 -9
- intentkit/skills/cryptocompare/__init__.py +7 -24
- intentkit/skills/cryptocompare/api.py +2 -3
- intentkit/skills/cryptocompare/base.py +11 -25
- intentkit/skills/cryptocompare/fetch_news.py +4 -5
- intentkit/skills/cryptocompare/fetch_price.py +6 -7
- intentkit/skills/cryptocompare/fetch_top_exchanges.py +4 -5
- intentkit/skills/cryptocompare/fetch_top_market_cap.py +4 -5
- intentkit/skills/cryptocompare/fetch_top_volume.py +4 -5
- intentkit/skills/cryptocompare/fetch_trading_signals.py +5 -6
- intentkit/skills/cryptocompare/schema.json +3 -3
- intentkit/skills/cryptopanic/__init__.py +7 -10
- intentkit/skills/cryptopanic/base.py +51 -55
- intentkit/skills/cryptopanic/fetch_crypto_news.py +4 -8
- intentkit/skills/cryptopanic/fetch_crypto_sentiment.py +5 -7
- intentkit/skills/cryptopanic/schema.json +105 -103
- intentkit/skills/dapplooker/__init__.py +2 -9
- intentkit/skills/dapplooker/base.py +4 -9
- intentkit/skills/dapplooker/dapplooker_token_data.py +7 -7
- intentkit/skills/dapplooker/schema.json +3 -5
- intentkit/skills/defillama/__init__.py +24 -74
- intentkit/skills/defillama/api.py +6 -9
- intentkit/skills/defillama/base.py +11 -21
- intentkit/skills/defillama/coins/fetch_batch_historical_prices.py +8 -10
- intentkit/skills/defillama/coins/fetch_block.py +6 -8
- intentkit/skills/defillama/coins/fetch_current_prices.py +8 -10
- intentkit/skills/defillama/coins/fetch_first_price.py +7 -9
- intentkit/skills/defillama/coins/fetch_historical_prices.py +9 -11
- intentkit/skills/defillama/coins/fetch_price_chart.py +9 -11
- intentkit/skills/defillama/coins/fetch_price_percentage.py +7 -9
- intentkit/skills/defillama/config/chains.py +1 -3
- intentkit/skills/defillama/fees/fetch_fees_overview.py +24 -26
- intentkit/skills/defillama/schema.json +5 -1
- intentkit/skills/defillama/stablecoins/fetch_stablecoin_chains.py +16 -18
- intentkit/skills/defillama/stablecoins/fetch_stablecoin_charts.py +8 -10
- intentkit/skills/defillama/stablecoins/fetch_stablecoin_prices.py +5 -7
- intentkit/skills/defillama/stablecoins/fetch_stablecoins.py +7 -9
- intentkit/skills/defillama/tests/api_integration.test.py +1 -1
- intentkit/skills/defillama/tvl/fetch_chain_historical_tvl.py +4 -6
- intentkit/skills/defillama/tvl/fetch_chains.py +9 -11
- intentkit/skills/defillama/tvl/fetch_historical_tvl.py +4 -6
- intentkit/skills/defillama/tvl/fetch_protocol.py +32 -38
- intentkit/skills/defillama/tvl/fetch_protocol_current_tvl.py +3 -5
- intentkit/skills/defillama/tvl/fetch_protocols.py +37 -45
- intentkit/skills/defillama/volumes/fetch_dex_overview.py +42 -48
- intentkit/skills/defillama/volumes/fetch_dex_summary.py +35 -37
- intentkit/skills/defillama/volumes/fetch_options_overview.py +24 -28
- intentkit/skills/defillama/yields/fetch_pool_chart.py +10 -12
- intentkit/skills/defillama/yields/fetch_pools.py +26 -30
- intentkit/skills/dexscreener/README.md +154 -0
- intentkit/skills/dexscreener/__init__.py +97 -93
- intentkit/skills/dexscreener/base.py +125 -133
- intentkit/skills/dexscreener/get_pair_info.py +158 -0
- intentkit/skills/dexscreener/get_token_pairs.py +165 -0
- intentkit/skills/dexscreener/get_tokens_info.py +212 -0
- intentkit/skills/dexscreener/model/search_token_response.py +80 -82
- intentkit/skills/dexscreener/schema.json +91 -48
- intentkit/skills/dexscreener/search_token.py +182 -321
- intentkit/skills/dexscreener/utils.py +420 -0
- intentkit/skills/dune_analytics/__init__.py +7 -9
- intentkit/skills/dune_analytics/base.py +48 -52
- intentkit/skills/dune_analytics/fetch_kol_buys.py +5 -7
- intentkit/skills/dune_analytics/fetch_nation_metrics.py +6 -8
- intentkit/skills/dune_analytics/schema.json +104 -99
- intentkit/skills/elfa/__init__.py +5 -18
- intentkit/skills/elfa/base.py +10 -14
- intentkit/skills/elfa/mention.py +19 -21
- intentkit/skills/elfa/schema.json +3 -2
- intentkit/skills/elfa/stats.py +4 -4
- intentkit/skills/elfa/tokens.py +12 -12
- intentkit/skills/elfa/utils.py +26 -28
- intentkit/skills/enso/__init__.py +11 -31
- intentkit/skills/enso/base.py +50 -35
- intentkit/skills/enso/best_yield.py +16 -24
- intentkit/skills/enso/networks.py +6 -11
- intentkit/skills/enso/prices.py +11 -13
- intentkit/skills/enso/route.py +34 -38
- intentkit/skills/enso/schema.json +3 -2
- intentkit/skills/enso/tokens.py +29 -38
- intentkit/skills/enso/wallet.py +76 -191
- intentkit/skills/erc20/__init__.py +50 -0
- intentkit/skills/erc20/base.py +11 -0
- intentkit/skills/erc20/erc20.svg +5 -0
- intentkit/skills/erc20/schema.json +74 -0
- intentkit/skills/erc721/__init__.py +53 -0
- intentkit/skills/erc721/base.py +11 -0
- intentkit/skills/erc721/erc721.svg +5 -0
- intentkit/skills/erc721/schema.json +90 -0
- intentkit/skills/firecrawl/README.md +11 -5
- intentkit/skills/firecrawl/__init__.py +5 -18
- intentkit/skills/firecrawl/base.py +4 -11
- intentkit/skills/firecrawl/clear.py +4 -8
- intentkit/skills/firecrawl/crawl.py +19 -19
- intentkit/skills/firecrawl/query.py +4 -3
- intentkit/skills/firecrawl/schema.json +6 -8
- intentkit/skills/firecrawl/scrape.py +150 -40
- intentkit/skills/firecrawl/utils.py +50 -42
- intentkit/skills/github/__init__.py +2 -7
- intentkit/skills/github/base.py +1 -7
- intentkit/skills/github/github_search.py +1 -2
- intentkit/skills/github/schema.json +3 -4
- intentkit/skills/heurist/__init__.py +8 -27
- intentkit/skills/heurist/base.py +4 -9
- intentkit/skills/heurist/image_generation_animagine_xl.py +12 -13
- intentkit/skills/heurist/image_generation_arthemy_comics.py +12 -13
- intentkit/skills/heurist/image_generation_arthemy_real.py +12 -13
- intentkit/skills/heurist/image_generation_braindance.py +12 -13
- intentkit/skills/heurist/image_generation_cyber_realistic_xl.py +12 -13
- intentkit/skills/heurist/image_generation_flux_1_dev.py +12 -13
- intentkit/skills/heurist/image_generation_sdxl.py +12 -13
- intentkit/skills/heurist/schema.json +2 -2
- intentkit/skills/http/__init__.py +4 -15
- intentkit/skills/http/base.py +1 -7
- intentkit/skills/http/get.py +21 -16
- intentkit/skills/http/post.py +23 -18
- intentkit/skills/http/put.py +23 -18
- intentkit/skills/http/schema.json +4 -5
- intentkit/skills/lifi/__init__.py +8 -13
- intentkit/skills/lifi/base.py +1 -7
- intentkit/skills/lifi/schema.json +17 -8
- intentkit/skills/lifi/token_execute.py +36 -30
- intentkit/skills/lifi/token_quote.py +8 -10
- intentkit/skills/lifi/utils.py +104 -51
- intentkit/skills/moralis/__init__.py +6 -10
- intentkit/skills/moralis/api.py +6 -7
- intentkit/skills/moralis/base.py +5 -10
- 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/moralis/schema.json +7 -2
- intentkit/skills/morpho/__init__.py +52 -0
- intentkit/skills/morpho/base.py +11 -0
- intentkit/skills/morpho/morpho.svg +12 -0
- intentkit/skills/morpho/schema.json +73 -0
- intentkit/skills/nation/__init__.py +4 -9
- intentkit/skills/nation/base.py +5 -10
- intentkit/skills/nation/nft_check.py +3 -4
- intentkit/skills/nation/schema.json +4 -3
- intentkit/skills/onchain.py +23 -0
- intentkit/skills/openai/__init__.py +17 -18
- intentkit/skills/openai/base.py +10 -14
- intentkit/skills/openai/dalle_image_generation.py +3 -8
- intentkit/skills/openai/gpt_avatar_generator.py +102 -0
- intentkit/skills/openai/gpt_image_generation.py +4 -8
- intentkit/skills/openai/gpt_image_mini_generator.py +91 -0
- intentkit/skills/openai/gpt_image_to_image.py +4 -8
- intentkit/skills/openai/image_to_text.py +3 -7
- intentkit/skills/openai/schema.json +34 -3
- intentkit/skills/portfolio/__init__.py +11 -35
- intentkit/skills/portfolio/base.py +33 -19
- intentkit/skills/portfolio/schema.json +3 -5
- intentkit/skills/portfolio/token_balances.py +21 -21
- intentkit/skills/portfolio/wallet_approvals.py +17 -18
- intentkit/skills/portfolio/wallet_defi_positions.py +3 -3
- intentkit/skills/portfolio/wallet_history.py +31 -31
- intentkit/skills/portfolio/wallet_net_worth.py +13 -13
- intentkit/skills/portfolio/wallet_nfts.py +19 -19
- intentkit/skills/portfolio/wallet_profitability.py +18 -18
- 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 +50 -0
- intentkit/skills/pyth/base.py +11 -0
- intentkit/skills/pyth/pyth.svg +6 -0
- intentkit/skills/pyth/schema.json +75 -0
- intentkit/skills/skills.toml +40 -0
- intentkit/skills/slack/__init__.py +5 -17
- intentkit/skills/slack/base.py +3 -9
- 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/schema.json +2 -2
- intentkit/skills/slack/send_message.py +3 -5
- intentkit/skills/supabase/__init__.py +7 -23
- intentkit/skills/supabase/base.py +9 -13
- intentkit/skills/supabase/delete_data.py +5 -6
- intentkit/skills/supabase/fetch_data.py +13 -14
- intentkit/skills/supabase/insert_data.py +5 -6
- intentkit/skills/supabase/invoke_function.py +7 -8
- intentkit/skills/supabase/schema.json +2 -3
- intentkit/skills/supabase/update_data.py +7 -8
- intentkit/skills/supabase/upsert_data.py +5 -6
- intentkit/skills/superfluid/__init__.py +53 -0
- intentkit/skills/superfluid/base.py +11 -0
- intentkit/skills/superfluid/schema.json +89 -0
- intentkit/skills/superfluid/superfluid.svg +6 -0
- intentkit/skills/system/__init__.py +7 -24
- intentkit/skills/system/add_autonomous_task.py +10 -12
- intentkit/skills/system/delete_autonomous_task.py +2 -2
- intentkit/skills/system/edit_autonomous_task.py +14 -18
- intentkit/skills/system/list_autonomous_tasks.py +3 -5
- intentkit/skills/system/read_agent_api_key.py +6 -4
- intentkit/skills/system/regenerate_agent_api_key.py +6 -4
- intentkit/skills/system/schema.json +6 -8
- intentkit/skills/tavily/__init__.py +3 -12
- intentkit/skills/tavily/base.py +4 -9
- intentkit/skills/tavily/schema.json +3 -5
- intentkit/skills/tavily/tavily_extract.py +2 -4
- intentkit/skills/tavily/tavily_search.py +4 -6
- intentkit/skills/token/__init__.py +5 -10
- intentkit/skills/token/base.py +7 -11
- intentkit/skills/token/erc20_transfers.py +19 -19
- intentkit/skills/token/schema.json +3 -6
- 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/__init__.py +11 -35
- intentkit/skills/twitter/base.py +23 -35
- intentkit/skills/twitter/follow_user.py +3 -7
- intentkit/skills/twitter/get_mentions.py +6 -13
- intentkit/skills/twitter/get_timeline.py +5 -13
- intentkit/skills/twitter/get_user_by_username.py +3 -7
- intentkit/skills/twitter/get_user_tweets.py +6 -14
- intentkit/skills/twitter/like_tweet.py +3 -7
- intentkit/skills/twitter/post_tweet.py +23 -12
- intentkit/skills/twitter/reply_tweet.py +21 -12
- intentkit/skills/twitter/retweet.py +3 -7
- intentkit/skills/twitter/schema.json +1 -0
- intentkit/skills/twitter/search_tweets.py +5 -13
- intentkit/skills/unrealspeech/__init__.py +2 -7
- intentkit/skills/unrealspeech/base.py +2 -8
- intentkit/skills/unrealspeech/schema.json +2 -5
- intentkit/skills/unrealspeech/text_to_speech.py +8 -8
- intentkit/skills/venice_audio/__init__.py +98 -106
- intentkit/skills/venice_audio/base.py +117 -121
- intentkit/skills/venice_audio/input.py +41 -41
- intentkit/skills/venice_audio/schema.json +151 -152
- intentkit/skills/venice_audio/venice_audio.py +38 -21
- intentkit/skills/venice_image/__init__.py +147 -154
- intentkit/skills/venice_image/api.py +138 -138
- intentkit/skills/venice_image/base.py +185 -192
- 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/schema.json +267 -267
- intentkit/skills/venice_image/utils.py +77 -78
- intentkit/skills/web_scraper/__init__.py +5 -18
- intentkit/skills/web_scraper/base.py +21 -7
- intentkit/skills/web_scraper/document_indexer.py +7 -6
- intentkit/skills/web_scraper/schema.json +2 -6
- intentkit/skills/web_scraper/scrape_and_index.py +15 -15
- intentkit/skills/web_scraper/utils.py +62 -63
- intentkit/skills/web_scraper/website_indexer.py +17 -19
- intentkit/skills/weth/__init__.py +49 -0
- intentkit/skills/weth/base.py +11 -0
- intentkit/skills/weth/schema.json +58 -0
- intentkit/skills/weth/weth.svg +6 -0
- intentkit/skills/wow/__init__.py +51 -0
- intentkit/skills/wow/base.py +11 -0
- intentkit/skills/wow/schema.json +89 -0
- intentkit/skills/wow/wow.svg +7 -0
- intentkit/skills/x402/__init__.py +61 -0
- intentkit/skills/x402/ask_agent.py +98 -0
- intentkit/skills/x402/base.py +99 -0
- intentkit/skills/x402/http_request.py +117 -0
- intentkit/skills/x402/schema.json +45 -0
- intentkit/skills/x402/x402.webp +0 -0
- intentkit/skills/xmtp/__init__.py +4 -15
- intentkit/skills/xmtp/base.py +61 -2
- intentkit/skills/xmtp/price.py +18 -13
- intentkit/skills/xmtp/schema.json +69 -71
- intentkit/skills/xmtp/swap.py +22 -25
- intentkit/skills/xmtp/transfer.py +71 -32
- intentkit/utils/chain.py +3 -3
- intentkit/utils/error.py +14 -1
- intentkit/utils/logging.py +2 -4
- intentkit/utils/s3.py +59 -7
- intentkit/utils/schema.py +100 -0
- intentkit/utils/slack_alert.py +7 -8
- {intentkit-0.6.13.dev2.dist-info → intentkit-0.8.17.dist-info}/METADATA +14 -16
- intentkit-0.8.17.dist-info/RECORD +466 -0
- intentkit/abstracts/exception.py +0 -9
- intentkit/core/skill.py +0 -200
- intentkit/models/generator.py +0 -347
- intentkit/skills/cdp/get_balance.py +0 -110
- intentkit/skills/cdp/swap.py +0 -121
- intentkit/skills/moralis/tests/__init__.py +0 -0
- intentkit/skills/moralis/tests/test_wallet.py +0 -511
- intentkit-0.6.13.dev2.dist-info/RECORD +0 -409
- {intentkit-0.6.13.dev2.dist-info → intentkit-0.8.17.dist-info}/WHEEL +0 -0
- {intentkit-0.6.13.dev2.dist-info → intentkit-0.8.17.dist-info}/licenses/LICENSE +0 -0
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
{
|
|
2
|
+
"$schema": "http://json-schema.org/draft-07/schema#",
|
|
3
|
+
"type": "object",
|
|
4
|
+
"title": "WOW",
|
|
5
|
+
"description": "WOW social token actions via Coinbase AgentKit",
|
|
6
|
+
"x-icon": "https://ai.service.crestal.dev/skills/wow/wow.svg",
|
|
7
|
+
"x-tags": [
|
|
8
|
+
"Crypto"
|
|
9
|
+
],
|
|
10
|
+
"properties": {
|
|
11
|
+
"enabled": {
|
|
12
|
+
"type": "boolean",
|
|
13
|
+
"title": "Enabled",
|
|
14
|
+
"description": "Whether this skill is enabled",
|
|
15
|
+
"default": false
|
|
16
|
+
},
|
|
17
|
+
"states": {
|
|
18
|
+
"type": "object",
|
|
19
|
+
"properties": {
|
|
20
|
+
"WowActionProvider_buy_token": {
|
|
21
|
+
"type": "string",
|
|
22
|
+
"title": "Buy Token",
|
|
23
|
+
"enum": [
|
|
24
|
+
"disabled",
|
|
25
|
+
"public",
|
|
26
|
+
"private"
|
|
27
|
+
],
|
|
28
|
+
"x-enum-title": [
|
|
29
|
+
"Disabled",
|
|
30
|
+
"Agent Owner + All Users",
|
|
31
|
+
"Agent Owner Only"
|
|
32
|
+
],
|
|
33
|
+
"description": "State for WowActionProvider_buy_token",
|
|
34
|
+
"default": "disabled"
|
|
35
|
+
},
|
|
36
|
+
"WowActionProvider_create_token": {
|
|
37
|
+
"type": "string",
|
|
38
|
+
"title": "Create Token",
|
|
39
|
+
"enum": [
|
|
40
|
+
"disabled",
|
|
41
|
+
"public",
|
|
42
|
+
"private"
|
|
43
|
+
],
|
|
44
|
+
"x-enum-title": [
|
|
45
|
+
"Disabled",
|
|
46
|
+
"Agent Owner + All Users",
|
|
47
|
+
"Agent Owner Only"
|
|
48
|
+
],
|
|
49
|
+
"description": "State for WowActionProvider_create_token",
|
|
50
|
+
"default": "disabled"
|
|
51
|
+
},
|
|
52
|
+
"WowActionProvider_sell_token": {
|
|
53
|
+
"type": "string",
|
|
54
|
+
"title": "Sell Token",
|
|
55
|
+
"enum": [
|
|
56
|
+
"disabled",
|
|
57
|
+
"public",
|
|
58
|
+
"private"
|
|
59
|
+
],
|
|
60
|
+
"x-enum-title": [
|
|
61
|
+
"Disabled",
|
|
62
|
+
"Agent Owner + All Users",
|
|
63
|
+
"Agent Owner Only"
|
|
64
|
+
],
|
|
65
|
+
"description": "State for WowActionProvider_sell_token",
|
|
66
|
+
"default": "disabled"
|
|
67
|
+
}
|
|
68
|
+
},
|
|
69
|
+
"description": "States for each WOW skill (disabled, public, or private)"
|
|
70
|
+
},
|
|
71
|
+
"api_key_provider": {
|
|
72
|
+
"type": "string",
|
|
73
|
+
"title": "API Key Provider",
|
|
74
|
+
"description": "Who provides the API key",
|
|
75
|
+
"enum": [
|
|
76
|
+
"platform"
|
|
77
|
+
],
|
|
78
|
+
"x-enum-title": [
|
|
79
|
+
"Nation Hosted"
|
|
80
|
+
],
|
|
81
|
+
"default": "platform"
|
|
82
|
+
}
|
|
83
|
+
},
|
|
84
|
+
"required": [
|
|
85
|
+
"states",
|
|
86
|
+
"enabled"
|
|
87
|
+
],
|
|
88
|
+
"additionalProperties": true
|
|
89
|
+
}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 128 128">
|
|
2
|
+
<rect width="128" height="128" fill="#0f766e" rx="16" />
|
|
3
|
+
<text x="50%" y="50%" font-family="Arial,Helvetica,sans-serif" font-size="48" fill="#ecfeff" font-weight="700" text-anchor="middle">WOW</text>
|
|
4
|
+
<circle cx="32" cy="96" r="12" fill="#5eead4" />
|
|
5
|
+
<circle cx="64" cy="96" r="12" fill="#34d399" />
|
|
6
|
+
<circle cx="96" cy="96" r="12" fill="#22c55e" />
|
|
7
|
+
</svg>
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
"""x402 skill category."""
|
|
2
|
+
|
|
3
|
+
import logging
|
|
4
|
+
from typing import TypedDict
|
|
5
|
+
|
|
6
|
+
from intentkit.skills.base import SkillConfig, SkillState
|
|
7
|
+
from intentkit.skills.x402.ask_agent import X402AskAgent
|
|
8
|
+
from intentkit.skills.x402.base import X402BaseSkill
|
|
9
|
+
from intentkit.skills.x402.http_request import X402HttpRequest
|
|
10
|
+
|
|
11
|
+
logger = logging.getLogger(__name__)
|
|
12
|
+
|
|
13
|
+
_cache: dict[str, X402BaseSkill] = {}
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
class SkillStates(TypedDict):
|
|
17
|
+
x402_ask_agent: SkillState
|
|
18
|
+
x402_http_request: SkillState
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
class Config(SkillConfig):
|
|
22
|
+
"""Configuration for x402 skills."""
|
|
23
|
+
|
|
24
|
+
states: SkillStates
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
_SKILL_BUILDERS: dict[str, type[X402BaseSkill]] = {
|
|
28
|
+
"x402_ask_agent": X402AskAgent,
|
|
29
|
+
"x402_http_request": X402HttpRequest,
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
async def get_skills(
|
|
34
|
+
config: "Config",
|
|
35
|
+
is_private: bool,
|
|
36
|
+
**_,
|
|
37
|
+
) -> list[X402BaseSkill]:
|
|
38
|
+
"""Return enabled x402 skills for the agent."""
|
|
39
|
+
enabled_skills = []
|
|
40
|
+
for skill_name, state in config["states"].items():
|
|
41
|
+
if state == "disabled":
|
|
42
|
+
continue
|
|
43
|
+
if state == "public" or (state == "private" and is_private):
|
|
44
|
+
enabled_skills.append(skill_name)
|
|
45
|
+
|
|
46
|
+
result: list[X402BaseSkill] = []
|
|
47
|
+
for name in enabled_skills:
|
|
48
|
+
skill = _get_skill(name)
|
|
49
|
+
if skill:
|
|
50
|
+
result.append(skill)
|
|
51
|
+
return result
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
def _get_skill(name: str) -> X402BaseSkill | None:
|
|
55
|
+
builder = _SKILL_BUILDERS.get(name)
|
|
56
|
+
if builder:
|
|
57
|
+
if name not in _cache:
|
|
58
|
+
_cache[name] = builder()
|
|
59
|
+
return _cache[name]
|
|
60
|
+
logger.warning("Unknown x402 skill requested: %s", name)
|
|
61
|
+
return None
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
from typing import Any
|
|
3
|
+
|
|
4
|
+
from langchain_core.tools import ToolException
|
|
5
|
+
from pydantic import BaseModel, Field
|
|
6
|
+
|
|
7
|
+
from intentkit.config.config import config
|
|
8
|
+
from intentkit.models.chat import AuthorType
|
|
9
|
+
from intentkit.skills.x402.base import X402BaseSkill
|
|
10
|
+
|
|
11
|
+
logger = logging.getLogger(__name__)
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class AskAgentInput(BaseModel):
|
|
15
|
+
"""Arguments for the x402 ask agent skill."""
|
|
16
|
+
|
|
17
|
+
agent_id: str = Field(description="ID or slug of the agent to query.")
|
|
18
|
+
message: str = Field(description="Message to send to the target agent.")
|
|
19
|
+
search_mode: bool | None = Field(
|
|
20
|
+
default=None, description="Enable search mode when interacting with the agent."
|
|
21
|
+
)
|
|
22
|
+
super_mode: bool | None = Field(
|
|
23
|
+
default=None, description="Enable super mode when interacting with the agent."
|
|
24
|
+
)
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
class X402AskAgent(X402BaseSkill):
|
|
28
|
+
"""Skill that queries another agent via the x402 API."""
|
|
29
|
+
|
|
30
|
+
name: str = "x402_ask_agent"
|
|
31
|
+
description: str = (
|
|
32
|
+
"Call another agent through the x402 API and return the final agent message."
|
|
33
|
+
)
|
|
34
|
+
args_schema: type[BaseModel] = AskAgentInput
|
|
35
|
+
|
|
36
|
+
async def _arun(
|
|
37
|
+
self,
|
|
38
|
+
agent_id: str,
|
|
39
|
+
message: str,
|
|
40
|
+
search_mode: bool | None = None,
|
|
41
|
+
super_mode: bool | None = None,
|
|
42
|
+
) -> str:
|
|
43
|
+
try:
|
|
44
|
+
# Use wallet provider signer to satisfy eth_account.BaseAccount interface requirements
|
|
45
|
+
base_url = (config.open_api_base_url or "").rstrip("/")
|
|
46
|
+
if not base_url:
|
|
47
|
+
raise ToolException("X402 API base URL is not configured.")
|
|
48
|
+
target_url = f"{base_url}/x402"
|
|
49
|
+
payload: dict[str, Any] = {
|
|
50
|
+
"agent_id": agent_id,
|
|
51
|
+
"message": message,
|
|
52
|
+
"app_id": "skill",
|
|
53
|
+
}
|
|
54
|
+
if search_mode is not None:
|
|
55
|
+
payload["search_mode"] = search_mode
|
|
56
|
+
if super_mode is not None:
|
|
57
|
+
payload["super_mode"] = super_mode
|
|
58
|
+
|
|
59
|
+
async with self.http_client(timeout=20.0) as client:
|
|
60
|
+
response = await client.post(target_url, json=payload)
|
|
61
|
+
try:
|
|
62
|
+
response.raise_for_status()
|
|
63
|
+
except Exception as e:
|
|
64
|
+
error_body = ""
|
|
65
|
+
try:
|
|
66
|
+
error_body = response.text
|
|
67
|
+
except Exception:
|
|
68
|
+
error_body = "Unable to read response body"
|
|
69
|
+
logger.error(
|
|
70
|
+
f"HTTP request failed with status {response.status_code}: {error_body}"
|
|
71
|
+
)
|
|
72
|
+
raise ToolException(
|
|
73
|
+
f"HTTP request failed with status {response.status_code}: {error_body}"
|
|
74
|
+
) from e
|
|
75
|
+
messages = response.json()
|
|
76
|
+
if not isinstance(messages, list) or not messages:
|
|
77
|
+
raise ValueError("Agent returned an empty response.")
|
|
78
|
+
|
|
79
|
+
last_message = messages[-1]
|
|
80
|
+
if not isinstance(last_message, dict):
|
|
81
|
+
raise ValueError("Agent response format is invalid.")
|
|
82
|
+
|
|
83
|
+
author_type = last_message.get("author_type")
|
|
84
|
+
content = last_message.get("message")
|
|
85
|
+
|
|
86
|
+
if author_type == AuthorType.SYSTEM.value:
|
|
87
|
+
raise ToolException(content or "Agent returned a system message.")
|
|
88
|
+
|
|
89
|
+
if not content:
|
|
90
|
+
raise ToolException("Agent response did not include message text.")
|
|
91
|
+
|
|
92
|
+
return str(content)
|
|
93
|
+
except ToolException:
|
|
94
|
+
# Re-raise ToolException as-is
|
|
95
|
+
raise
|
|
96
|
+
except Exception as e:
|
|
97
|
+
logger.error(f"Unexpected error in x402_ask_agent: {str(e)}")
|
|
98
|
+
raise ToolException(f"Unexpected error occurred: {str(e)}") from e
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
import threading
|
|
3
|
+
from collections.abc import AsyncIterator
|
|
4
|
+
from contextlib import asynccontextmanager
|
|
5
|
+
from typing import Any
|
|
6
|
+
|
|
7
|
+
from coinbase_agentkit.wallet_providers.evm_wallet_provider import (
|
|
8
|
+
EvmWalletSigner as CoinbaseEvmWalletSigner,
|
|
9
|
+
)
|
|
10
|
+
from x402.clients.httpx import x402HttpxClient
|
|
11
|
+
|
|
12
|
+
from intentkit.clients import get_wallet_provider
|
|
13
|
+
from intentkit.skills.onchain import IntentKitOnChainSkill
|
|
14
|
+
|
|
15
|
+
logger = logging.getLogger(__name__)
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
class X402BaseSkill(IntentKitOnChainSkill):
|
|
19
|
+
"""Base class for x402 skills."""
|
|
20
|
+
|
|
21
|
+
@property
|
|
22
|
+
def category(self) -> str:
|
|
23
|
+
return "x402"
|
|
24
|
+
|
|
25
|
+
async def _get_signer(self) -> "ThreadSafeEvmWalletSigner":
|
|
26
|
+
context = self.get_context()
|
|
27
|
+
wallet_provider = await get_wallet_provider(context.agent)
|
|
28
|
+
return ThreadSafeEvmWalletSigner(wallet_provider)
|
|
29
|
+
|
|
30
|
+
@asynccontextmanager
|
|
31
|
+
async def http_client(
|
|
32
|
+
self,
|
|
33
|
+
timeout: float = 30.0,
|
|
34
|
+
) -> AsyncIterator[x402HttpxClient]:
|
|
35
|
+
account = await self._get_signer()
|
|
36
|
+
try:
|
|
37
|
+
async with x402HttpxClient(
|
|
38
|
+
account=account,
|
|
39
|
+
timeout=timeout,
|
|
40
|
+
) as client:
|
|
41
|
+
yield client
|
|
42
|
+
except Exception:
|
|
43
|
+
logger.exception("Failed to create x402 HTTP client")
|
|
44
|
+
raise
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
class ThreadSafeEvmWalletSigner(CoinbaseEvmWalletSigner):
|
|
48
|
+
"""EVM wallet signer that avoids nested event loop errors.
|
|
49
|
+
|
|
50
|
+
Coinbase's signer runs async wallet calls in the current thread. When invoked
|
|
51
|
+
inside an active asyncio loop (as happens in async skills), it trips over the
|
|
52
|
+
loop already running. We hop work to a background thread so the provider can
|
|
53
|
+
spin up its own loop safely.
|
|
54
|
+
"""
|
|
55
|
+
|
|
56
|
+
def __init__(self, wallet_provider: Any):
|
|
57
|
+
super().__init__(wallet_provider=wallet_provider)
|
|
58
|
+
|
|
59
|
+
def _run_in_thread(self, func: Any, *args: Any, **kwargs: Any) -> Any:
|
|
60
|
+
result: list[Any] = []
|
|
61
|
+
error: list[BaseException] = []
|
|
62
|
+
|
|
63
|
+
def _target() -> None:
|
|
64
|
+
try:
|
|
65
|
+
result.append(func(*args, **kwargs))
|
|
66
|
+
except BaseException as exc: # pragma: no cover - bubble up original error
|
|
67
|
+
error.append(exc)
|
|
68
|
+
|
|
69
|
+
thread = threading.Thread(target=_target, daemon=True)
|
|
70
|
+
thread.start()
|
|
71
|
+
thread.join()
|
|
72
|
+
|
|
73
|
+
if error:
|
|
74
|
+
raise error[0]
|
|
75
|
+
return result[0] if result else None
|
|
76
|
+
|
|
77
|
+
def unsafe_sign_hash(self, message_hash: Any) -> Any:
|
|
78
|
+
return self._run_in_thread(super().unsafe_sign_hash, message_hash)
|
|
79
|
+
|
|
80
|
+
def sign_message(self, signable_message: Any) -> Any:
|
|
81
|
+
return self._run_in_thread(super().sign_message, signable_message)
|
|
82
|
+
|
|
83
|
+
def sign_transaction(self, transaction_dict: Any) -> Any:
|
|
84
|
+
return self._run_in_thread(super().sign_transaction, transaction_dict)
|
|
85
|
+
|
|
86
|
+
def sign_typed_data(
|
|
87
|
+
self,
|
|
88
|
+
domain_data: Any | None = None,
|
|
89
|
+
message_types: Any | None = None,
|
|
90
|
+
message_data: Any | None = None,
|
|
91
|
+
full_message: Any | None = None,
|
|
92
|
+
) -> Any:
|
|
93
|
+
return self._run_in_thread(
|
|
94
|
+
super().sign_typed_data,
|
|
95
|
+
domain_data=domain_data,
|
|
96
|
+
message_types=message_types,
|
|
97
|
+
message_data=message_data,
|
|
98
|
+
full_message=full_message,
|
|
99
|
+
)
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
from typing import Any
|
|
3
|
+
from urllib.parse import urlparse
|
|
4
|
+
|
|
5
|
+
import httpx
|
|
6
|
+
from langchain_core.tools import ToolException
|
|
7
|
+
from pydantic import BaseModel, Field
|
|
8
|
+
|
|
9
|
+
from intentkit.skills.x402.base import X402BaseSkill
|
|
10
|
+
|
|
11
|
+
logger = logging.getLogger(__name__)
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class X402HttpRequestInput(BaseModel):
|
|
15
|
+
"""Arguments for a generic x402 HTTP request."""
|
|
16
|
+
|
|
17
|
+
method: str = Field(description="HTTP method to use. Supported values: GET, POST.")
|
|
18
|
+
url: str = Field(
|
|
19
|
+
description="Absolute URL for the request (must include scheme and host)."
|
|
20
|
+
)
|
|
21
|
+
headers: dict[str, str] | None = Field(
|
|
22
|
+
default=None,
|
|
23
|
+
description="Optional headers to include in the request.",
|
|
24
|
+
)
|
|
25
|
+
params: dict[str, Any] | None = Field(
|
|
26
|
+
default=None,
|
|
27
|
+
description="Optional query parameters to include in the request.",
|
|
28
|
+
)
|
|
29
|
+
data: dict[str, Any] | str | None = Field(
|
|
30
|
+
default=None,
|
|
31
|
+
description=(
|
|
32
|
+
"Optional request body. Dictionaries are sent as JSON; strings are sent as raw data. "
|
|
33
|
+
"Only supported for POST requests."
|
|
34
|
+
),
|
|
35
|
+
)
|
|
36
|
+
timeout: float | None = Field(
|
|
37
|
+
default=30.0,
|
|
38
|
+
description="Request timeout in seconds.",
|
|
39
|
+
)
|
|
40
|
+
|
|
41
|
+
|
|
42
|
+
class X402HttpRequest(X402BaseSkill):
|
|
43
|
+
"""Skill that performs signed HTTP requests via the x402 client."""
|
|
44
|
+
|
|
45
|
+
name: str = "x402_http_request"
|
|
46
|
+
description: str = (
|
|
47
|
+
"Send an HTTP GET or POST request using the x402 payment protocol. "
|
|
48
|
+
"Provide the method, absolute URL, optional headers, query parameters, and request body. "
|
|
49
|
+
"Returns the response status and body text."
|
|
50
|
+
)
|
|
51
|
+
args_schema: type[BaseModel] = X402HttpRequestInput
|
|
52
|
+
|
|
53
|
+
async def _arun(
|
|
54
|
+
self,
|
|
55
|
+
method: str,
|
|
56
|
+
url: str,
|
|
57
|
+
headers: dict[str, str] | None = None,
|
|
58
|
+
params: dict[str, Any] | None = None,
|
|
59
|
+
data: dict[str, Any] | str | None = None,
|
|
60
|
+
timeout: float = 30.0,
|
|
61
|
+
**_: Any,
|
|
62
|
+
) -> str:
|
|
63
|
+
method_upper = method.upper()
|
|
64
|
+
if method_upper not in {"GET", "POST"}:
|
|
65
|
+
raise ToolException(
|
|
66
|
+
f"Unsupported HTTP method '{method}'. Only GET and POST are allowed."
|
|
67
|
+
)
|
|
68
|
+
|
|
69
|
+
parsed = urlparse(url)
|
|
70
|
+
if not (parsed.scheme and parsed.netloc):
|
|
71
|
+
raise ToolException("URL must include scheme and host (absolute URL).")
|
|
72
|
+
|
|
73
|
+
request_headers = dict(headers or {})
|
|
74
|
+
request_kwargs: dict[str, Any] = {
|
|
75
|
+
"url": url,
|
|
76
|
+
"headers": request_headers or None,
|
|
77
|
+
"params": params,
|
|
78
|
+
"timeout": timeout,
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
if method_upper == "POST":
|
|
82
|
+
if isinstance(data, dict):
|
|
83
|
+
header_keys = {key.lower() for key in request_headers}
|
|
84
|
+
if "content-type" not in header_keys:
|
|
85
|
+
request_headers["Content-Type"] = "application/json"
|
|
86
|
+
request_kwargs["json"] = data
|
|
87
|
+
elif isinstance(data, str):
|
|
88
|
+
request_kwargs["content"] = data
|
|
89
|
+
elif data is not None:
|
|
90
|
+
raise ToolException(
|
|
91
|
+
"POST body must be either a JSON-serializable object or a string."
|
|
92
|
+
)
|
|
93
|
+
elif data is not None:
|
|
94
|
+
raise ToolException("Request body is only supported for POST requests.")
|
|
95
|
+
|
|
96
|
+
try:
|
|
97
|
+
async with self.http_client(timeout=timeout) as client:
|
|
98
|
+
response = await client.request(method_upper, **request_kwargs)
|
|
99
|
+
response.raise_for_status()
|
|
100
|
+
return f"Status: {response.status_code}\nContent: {response.text}"
|
|
101
|
+
except ValueError as exc:
|
|
102
|
+
raise ToolException(str(exc)) from exc
|
|
103
|
+
except httpx.TimeoutException as exc:
|
|
104
|
+
raise ToolException(
|
|
105
|
+
f"Request to {url} timed out after {timeout} seconds"
|
|
106
|
+
) from exc
|
|
107
|
+
except httpx.HTTPStatusError as exc:
|
|
108
|
+
raise ToolException(
|
|
109
|
+
f"HTTP {exc.response.status_code} - {exc.response.text}"
|
|
110
|
+
) from exc
|
|
111
|
+
except httpx.RequestError as exc:
|
|
112
|
+
raise ToolException(f"Failed to connect to {url} - {str(exc)}") from exc
|
|
113
|
+
except ToolException:
|
|
114
|
+
raise
|
|
115
|
+
except Exception as exc:
|
|
116
|
+
logger.error("Unexpected error in x402_http_request", exc_info=exc)
|
|
117
|
+
raise ToolException(f"Unexpected error occurred - {str(exc)}") from exc
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
{
|
|
2
|
+
"$schema": "http://json-schema.org/draft-07/schema#",
|
|
3
|
+
"type": "object",
|
|
4
|
+
"title": "x402",
|
|
5
|
+
"description": "Interact with other IntentKit agents through the x402 payment protocol.",
|
|
6
|
+
"x-icon": "https://ai.service.crestal.dev/skills/x402/x402.webp",
|
|
7
|
+
"x-tags": ["Communication", "Infrastructure"],
|
|
8
|
+
"properties": {
|
|
9
|
+
"enabled": {
|
|
10
|
+
"type": "boolean",
|
|
11
|
+
"title": "Enabled",
|
|
12
|
+
"description": "Whether this skill category is enabled.",
|
|
13
|
+
"default": false
|
|
14
|
+
},
|
|
15
|
+
"states": {
|
|
16
|
+
"type": "object",
|
|
17
|
+
"properties": {
|
|
18
|
+
"x402_ask_agent": {
|
|
19
|
+
"type": "string",
|
|
20
|
+
"title": "Ask Agent",
|
|
21
|
+
"enum": ["disabled", "public", "private"],
|
|
22
|
+
"x-enum-title": [
|
|
23
|
+
"Disabled",
|
|
24
|
+
"Agent Owner + All Users",
|
|
25
|
+
"Agent Owner Only"
|
|
26
|
+
],
|
|
27
|
+
"description": "Send a message to another IntentKit agent via the x402 API and return the final agent response.",
|
|
28
|
+
"default": "disabled"
|
|
29
|
+
},
|
|
30
|
+
"x402_http_request": {
|
|
31
|
+
"type": "string",
|
|
32
|
+
"title": "HTTP Request",
|
|
33
|
+
"enum": ["disabled", "public", "private"],
|
|
34
|
+
"x-enum-title": [
|
|
35
|
+
"Disabled",
|
|
36
|
+
"Agent Owner + All Users",
|
|
37
|
+
"Agent Owner Only"
|
|
38
|
+
],
|
|
39
|
+
"description": "Perform a signed HTTP GET or POST request via the x402 payment protocol.",
|
|
40
|
+
"default": "disabled"
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
}
|
|
Binary file
|
|
@@ -3,7 +3,6 @@
|
|
|
3
3
|
import logging
|
|
4
4
|
from typing import TypedDict
|
|
5
5
|
|
|
6
|
-
from intentkit.abstracts.skill import SkillStoreABC
|
|
7
6
|
from intentkit.skills.base import SkillConfig, SkillState
|
|
8
7
|
from intentkit.skills.xmtp.base import XmtpBaseTool
|
|
9
8
|
from intentkit.skills.xmtp.price import XmtpGetSwapPrice
|
|
@@ -31,7 +30,6 @@ class Config(SkillConfig):
|
|
|
31
30
|
async def get_skills(
|
|
32
31
|
config: "Config",
|
|
33
32
|
is_private: bool,
|
|
34
|
-
store: SkillStoreABC,
|
|
35
33
|
**_,
|
|
36
34
|
) -> list[XmtpBaseTool]:
|
|
37
35
|
"""Get all XMTP skills.
|
|
@@ -39,7 +37,6 @@ async def get_skills(
|
|
|
39
37
|
Args:
|
|
40
38
|
config: The configuration for XMTP skills.
|
|
41
39
|
is_private: Whether to include private skills.
|
|
42
|
-
store: The skill store for persisting data.
|
|
43
40
|
|
|
44
41
|
Returns:
|
|
45
42
|
A list of XMTP skills.
|
|
@@ -56,7 +53,7 @@ async def get_skills(
|
|
|
56
53
|
# Get each skill using the cached getter
|
|
57
54
|
result = []
|
|
58
55
|
for name in available_skills:
|
|
59
|
-
skill = get_xmtp_skill(name
|
|
56
|
+
skill = get_xmtp_skill(name)
|
|
60
57
|
if skill:
|
|
61
58
|
result.append(skill)
|
|
62
59
|
return result
|
|
@@ -64,34 +61,26 @@ async def get_skills(
|
|
|
64
61
|
|
|
65
62
|
def get_xmtp_skill(
|
|
66
63
|
name: str,
|
|
67
|
-
store: SkillStoreABC,
|
|
68
64
|
) -> XmtpBaseTool:
|
|
69
65
|
"""Get an XMTP skill by name.
|
|
70
66
|
|
|
71
67
|
Args:
|
|
72
68
|
name: The name of the skill to get
|
|
73
|
-
store: The skill store for persisting data
|
|
74
69
|
|
|
75
70
|
Returns:
|
|
76
71
|
The requested XMTP skill
|
|
77
72
|
"""
|
|
78
73
|
if name == "xmtp_transfer":
|
|
79
74
|
if name not in _cache:
|
|
80
|
-
_cache[name] = XmtpTransfer(
|
|
81
|
-
skill_store=store,
|
|
82
|
-
)
|
|
75
|
+
_cache[name] = XmtpTransfer()
|
|
83
76
|
return _cache[name]
|
|
84
77
|
elif name == "xmtp_swap":
|
|
85
78
|
if name not in _cache:
|
|
86
|
-
_cache[name] = XmtpSwap(
|
|
87
|
-
skill_store=store,
|
|
88
|
-
)
|
|
79
|
+
_cache[name] = XmtpSwap()
|
|
89
80
|
return _cache[name]
|
|
90
81
|
elif name == "xmtp_get_swap_price":
|
|
91
82
|
if name not in _cache:
|
|
92
|
-
_cache[name] = XmtpGetSwapPrice(
|
|
93
|
-
skill_store=store,
|
|
94
|
-
)
|
|
83
|
+
_cache[name] = XmtpGetSwapPrice()
|
|
95
84
|
return _cache[name]
|
|
96
85
|
else:
|
|
97
86
|
logger.warning(f"Unknown XMTP skill: {name}")
|
intentkit/skills/xmtp/base.py
CHANGED
|
@@ -1,15 +1,74 @@
|
|
|
1
1
|
from typing import Literal
|
|
2
2
|
|
|
3
|
-
from intentkit.skills.
|
|
3
|
+
from intentkit.skills.onchain import IntentKitOnChainSkill
|
|
4
4
|
|
|
5
5
|
|
|
6
|
-
class XmtpBaseTool(
|
|
6
|
+
class XmtpBaseTool(IntentKitOnChainSkill):
|
|
7
7
|
"""Base class for XMTP-related skills."""
|
|
8
8
|
|
|
9
9
|
# Set response format to content_and_artifact for returning tuple
|
|
10
10
|
response_format: Literal["content", "content_and_artifact"] = "content_and_artifact"
|
|
11
11
|
|
|
12
|
+
# ChainId mapping for XMTP wallet_sendCalls (mainnet only)
|
|
13
|
+
CHAIN_ID_HEX_BY_NETWORK: dict[str, str] = {
|
|
14
|
+
"ethereum-mainnet": "0x1", # 1
|
|
15
|
+
"base-mainnet": "0x2105", # 8453
|
|
16
|
+
"arbitrum-mainnet": "0xA4B1", # 42161
|
|
17
|
+
"optimism-mainnet": "0xA", # 10
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
# CDP network mapping for swap quote API (mainnet only)
|
|
21
|
+
NETWORK_FOR_CDP_MAPPING: dict[str, str] = {
|
|
22
|
+
"ethereum-mainnet": "ethereum",
|
|
23
|
+
"base-mainnet": "base",
|
|
24
|
+
"arbitrum-mainnet": "arbitrum",
|
|
25
|
+
"optimism-mainnet": "optimism",
|
|
26
|
+
}
|
|
27
|
+
|
|
12
28
|
@property
|
|
13
29
|
def category(self) -> str:
|
|
14
30
|
"""Return the skill category."""
|
|
15
31
|
return "xmtp"
|
|
32
|
+
|
|
33
|
+
def validate_network_and_get_chain_id(
|
|
34
|
+
self, network_id: str, skill_name: str
|
|
35
|
+
) -> str:
|
|
36
|
+
"""Validate network and return chain ID hex.
|
|
37
|
+
|
|
38
|
+
Args:
|
|
39
|
+
network_id: The network ID to validate
|
|
40
|
+
skill_name: The name of the skill for error messages
|
|
41
|
+
|
|
42
|
+
Returns:
|
|
43
|
+
The hex chain ID for the network
|
|
44
|
+
|
|
45
|
+
Raises:
|
|
46
|
+
ValueError: If the network is not supported
|
|
47
|
+
"""
|
|
48
|
+
if network_id not in self.CHAIN_ID_HEX_BY_NETWORK:
|
|
49
|
+
supported_networks = ", ".join(self.CHAIN_ID_HEX_BY_NETWORK.keys())
|
|
50
|
+
raise ValueError(
|
|
51
|
+
f"XMTP {skill_name} supports the following networks: {supported_networks}. "
|
|
52
|
+
f"Current agent network: {network_id}"
|
|
53
|
+
)
|
|
54
|
+
return self.CHAIN_ID_HEX_BY_NETWORK[network_id]
|
|
55
|
+
|
|
56
|
+
def get_cdp_network(self, network_id: str) -> str:
|
|
57
|
+
"""Get CDP network name for the given network ID.
|
|
58
|
+
|
|
59
|
+
Args:
|
|
60
|
+
network_id: The network ID
|
|
61
|
+
|
|
62
|
+
Returns:
|
|
63
|
+
The CDP network name
|
|
64
|
+
|
|
65
|
+
Raises:
|
|
66
|
+
ValueError: If the network is not supported for CDP
|
|
67
|
+
"""
|
|
68
|
+
if network_id not in self.NETWORK_FOR_CDP_MAPPING:
|
|
69
|
+
supported_networks = ", ".join(self.NETWORK_FOR_CDP_MAPPING.keys())
|
|
70
|
+
raise ValueError(
|
|
71
|
+
f"CDP swap does not support network: {network_id}. "
|
|
72
|
+
f"Supported networks: {supported_networks}"
|
|
73
|
+
)
|
|
74
|
+
return self.NETWORK_FOR_CDP_MAPPING[network_id]
|