intentkit 0.8.12rc0__py3-none-any.whl → 0.8.13__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/skill.py +2 -59
- intentkit/clients/__init__.py +3 -2
- intentkit/clients/cdp.py +63 -44
- intentkit/clients/twitter.py +35 -28
- intentkit/config/config.py +1 -0
- intentkit/core/agent.py +2 -279
- intentkit/core/asset.py +63 -16
- intentkit/core/engine.py +9 -5
- intentkit/core/scheduler.py +8 -8
- intentkit/models/agent.py +138 -94
- intentkit/models/agent_schema.json +6 -9
- intentkit/models/chat.py +1 -0
- intentkit/models/llm.csv +15 -12
- intentkit/models/skills.csv +5 -4
- intentkit/skills/acolyt/__init__.py +2 -9
- intentkit/skills/acolyt/base.py +2 -5
- intentkit/skills/aixbt/__init__.py +2 -13
- intentkit/skills/aixbt/base.py +0 -4
- intentkit/skills/aixbt/projects.py +1 -2
- intentkit/skills/allora/__init__.py +2 -9
- intentkit/skills/allora/base.py +2 -5
- intentkit/skills/base.py +101 -37
- intentkit/skills/basename/__init__.py +1 -3
- intentkit/skills/carv/__init__.py +116 -121
- intentkit/skills/carv/base.py +184 -185
- intentkit/skills/casino/__init__.py +4 -15
- intentkit/skills/casino/base.py +0 -4
- 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 +0 -5
- intentkit/skills/cdp/base.py +0 -4
- intentkit/skills/cdp/schema.json +1 -17
- intentkit/skills/chainlist/__init__.py +2 -7
- intentkit/skills/chainlist/base.py +0 -4
- intentkit/skills/common/__init__.py +2 -9
- intentkit/skills/common/base.py +0 -4
- intentkit/skills/cookiefun/__init__.py +6 -9
- intentkit/skills/cookiefun/base.py +0 -4
- intentkit/skills/cryptocompare/__init__.py +7 -24
- intentkit/skills/cryptocompare/base.py +0 -5
- intentkit/skills/cryptopanic/__init__.py +3 -6
- intentkit/skills/cryptopanic/base.py +53 -55
- intentkit/skills/cryptopanic/fetch_crypto_news.py +0 -2
- intentkit/skills/cryptopanic/fetch_crypto_sentiment.py +1 -3
- intentkit/skills/dapplooker/__init__.py +2 -9
- intentkit/skills/dapplooker/base.py +2 -5
- intentkit/skills/defillama/__init__.py +24 -74
- intentkit/skills/defillama/base.py +0 -4
- intentkit/skills/defillama/coins/fetch_batch_historical_prices.py +2 -2
- intentkit/skills/defillama/coins/fetch_block.py +2 -2
- intentkit/skills/defillama/coins/fetch_current_prices.py +2 -2
- intentkit/skills/defillama/coins/fetch_first_price.py +2 -2
- intentkit/skills/defillama/coins/fetch_historical_prices.py +2 -2
- intentkit/skills/defillama/coins/fetch_price_chart.py +2 -2
- intentkit/skills/defillama/coins/fetch_price_percentage.py +2 -2
- intentkit/skills/defillama/fees/fetch_fees_overview.py +2 -2
- intentkit/skills/defillama/stablecoins/fetch_stablecoin_chains.py +2 -2
- intentkit/skills/defillama/stablecoins/fetch_stablecoin_charts.py +2 -2
- intentkit/skills/defillama/stablecoins/fetch_stablecoin_prices.py +2 -2
- intentkit/skills/defillama/stablecoins/fetch_stablecoins.py +2 -2
- intentkit/skills/defillama/tvl/fetch_chain_historical_tvl.py +2 -2
- intentkit/skills/defillama/tvl/fetch_chains.py +2 -2
- intentkit/skills/defillama/tvl/fetch_historical_tvl.py +2 -2
- intentkit/skills/defillama/tvl/fetch_protocol.py +2 -2
- intentkit/skills/defillama/tvl/fetch_protocol_current_tvl.py +2 -2
- intentkit/skills/defillama/tvl/fetch_protocols.py +2 -2
- intentkit/skills/defillama/volumes/fetch_dex_overview.py +2 -2
- intentkit/skills/defillama/volumes/fetch_dex_summary.py +2 -2
- intentkit/skills/defillama/volumes/fetch_options_overview.py +2 -2
- intentkit/skills/defillama/yields/fetch_pool_chart.py +2 -2
- intentkit/skills/defillama/yields/fetch_pools.py +2 -2
- intentkit/skills/dexscreener/__init__.py +97 -102
- intentkit/skills/dexscreener/base.py +125 -130
- intentkit/skills/dexscreener/get_pair_info.py +2 -3
- intentkit/skills/dexscreener/get_token_pairs.py +2 -3
- intentkit/skills/dexscreener/get_tokens_info.py +2 -3
- intentkit/skills/dexscreener/search_token.py +2 -4
- intentkit/skills/dune_analytics/__init__.py +4 -6
- intentkit/skills/dune_analytics/base.py +50 -52
- intentkit/skills/dune_analytics/fetch_kol_buys.py +0 -2
- intentkit/skills/dune_analytics/fetch_nation_metrics.py +0 -2
- intentkit/skills/elfa/__init__.py +5 -18
- intentkit/skills/elfa/base.py +8 -10
- intentkit/skills/enso/__init__.py +9 -29
- intentkit/skills/enso/base.py +3 -6
- intentkit/skills/enso/route.py +1 -3
- intentkit/skills/erc20/__init__.py +1 -5
- intentkit/skills/erc721/__init__.py +1 -3
- intentkit/skills/firecrawl/__init__.py +5 -18
- intentkit/skills/firecrawl/base.py +2 -5
- intentkit/skills/firecrawl/crawl.py +10 -9
- intentkit/skills/firecrawl/query.py +3 -1
- intentkit/skills/firecrawl/scrape.py +8 -10
- intentkit/skills/firecrawl/utils.py +25 -26
- intentkit/skills/github/__init__.py +2 -7
- intentkit/skills/github/base.py +0 -4
- intentkit/skills/heurist/__init__.py +8 -27
- intentkit/skills/heurist/base.py +2 -5
- intentkit/skills/heurist/image_generation_animagine_xl.py +5 -5
- intentkit/skills/heurist/image_generation_arthemy_comics.py +5 -5
- intentkit/skills/heurist/image_generation_arthemy_real.py +5 -5
- intentkit/skills/heurist/image_generation_braindance.py +5 -5
- intentkit/skills/heurist/image_generation_cyber_realistic_xl.py +5 -5
- intentkit/skills/heurist/image_generation_flux_1_dev.py +5 -5
- intentkit/skills/heurist/image_generation_sdxl.py +5 -5
- intentkit/skills/http/__init__.py +4 -15
- intentkit/skills/http/base.py +0 -4
- intentkit/skills/lifi/__init__.py +1 -6
- intentkit/skills/lifi/base.py +0 -4
- intentkit/skills/lifi/token_execute.py +1 -4
- intentkit/skills/lifi/token_quote.py +1 -3
- intentkit/skills/moralis/__init__.py +3 -7
- intentkit/skills/moralis/base.py +2 -5
- intentkit/skills/morpho/__init__.py +1 -3
- intentkit/skills/nation/__init__.py +2 -7
- intentkit/skills/nation/base.py +4 -7
- intentkit/skills/onchain.py +27 -0
- intentkit/skills/openai/__init__.py +5 -18
- intentkit/skills/openai/base.py +8 -10
- intentkit/skills/openai/dalle_image_generation.py +2 -5
- intentkit/skills/openai/gpt_image_generation.py +2 -5
- intentkit/skills/openai/gpt_image_to_image.py +2 -5
- intentkit/skills/openai/image_to_text.py +2 -5
- intentkit/skills/portfolio/__init__.py +11 -35
- intentkit/skills/portfolio/base.py +2 -5
- intentkit/skills/pyth/__init__.py +1 -5
- intentkit/skills/skills.toml +4 -0
- intentkit/skills/slack/__init__.py +5 -17
- intentkit/skills/slack/base.py +0 -4
- intentkit/skills/supabase/__init__.py +7 -23
- intentkit/skills/supabase/base.py +0 -4
- intentkit/skills/superfluid/__init__.py +1 -3
- intentkit/skills/system/__init__.py +7 -24
- intentkit/skills/system/add_autonomous_task.py +2 -2
- intentkit/skills/system/delete_autonomous_task.py +2 -2
- intentkit/skills/system/edit_autonomous_task.py +2 -4
- intentkit/skills/system/list_autonomous_tasks.py +2 -2
- intentkit/skills/system/read_agent_api_key.py +6 -4
- intentkit/skills/system/regenerate_agent_api_key.py +6 -4
- intentkit/skills/tavily/__init__.py +3 -12
- intentkit/skills/tavily/base.py +2 -5
- intentkit/skills/tavily/tavily_extract.py +1 -2
- intentkit/skills/tavily/tavily_search.py +3 -3
- intentkit/skills/token/__init__.py +5 -10
- intentkit/skills/token/base.py +2 -6
- intentkit/skills/twitter/__init__.py +11 -35
- intentkit/skills/twitter/base.py +14 -16
- intentkit/skills/twitter/follow_user.py +0 -1
- intentkit/skills/twitter/get_mentions.py +0 -1
- intentkit/skills/twitter/get_timeline.py +0 -1
- intentkit/skills/twitter/get_user_by_username.py +0 -1
- intentkit/skills/twitter/get_user_tweets.py +0 -1
- intentkit/skills/twitter/like_tweet.py +0 -1
- intentkit/skills/twitter/post_tweet.py +2 -2
- intentkit/skills/twitter/reply_tweet.py +2 -2
- intentkit/skills/twitter/retweet.py +0 -1
- intentkit/skills/twitter/search_tweets.py +0 -1
- intentkit/skills/unrealspeech/__init__.py +2 -7
- intentkit/skills/unrealspeech/base.py +0 -4
- intentkit/skills/venice_audio/__init__.py +99 -106
- intentkit/skills/venice_audio/base.py +118 -121
- intentkit/skills/venice_audio/venice_audio.py +1 -5
- intentkit/skills/venice_image/__init__.py +147 -154
- intentkit/skills/venice_image/base.py +185 -192
- intentkit/skills/web_scraper/__init__.py +5 -18
- intentkit/skills/web_scraper/base.py +20 -4
- intentkit/skills/web_scraper/document_indexer.py +6 -4
- intentkit/skills/web_scraper/scrape_and_index.py +11 -8
- intentkit/skills/web_scraper/utils.py +31 -27
- intentkit/skills/web_scraper/website_indexer.py +7 -8
- intentkit/skills/weth/__init__.py +1 -5
- intentkit/skills/wow/__init__.py +1 -5
- intentkit/skills/x402/__init__.py +53 -0
- intentkit/skills/x402/ask_agent.py +141 -0
- intentkit/skills/x402/base.py +9 -0
- intentkit/skills/x402/schema.json +40 -0
- intentkit/skills/x402/x402.png +0 -0
- intentkit/skills/xmtp/__init__.py +4 -15
- intentkit/skills/xmtp/base.py +2 -2
- intentkit/skills/xmtp/price.py +3 -3
- intentkit/skills/xmtp/swap.py +4 -4
- intentkit/utils/schema.py +100 -0
- {intentkit-0.8.12rc0.dist-info → intentkit-0.8.13.dist-info}/METADATA +1 -1
- {intentkit-0.8.12rc0.dist-info → intentkit-0.8.13.dist-info}/RECORD +188 -181
- {intentkit-0.8.12rc0.dist-info → intentkit-0.8.13.dist-info}/WHEEL +0 -0
- {intentkit-0.8.12rc0.dist-info → intentkit-0.8.13.dist-info}/licenses/LICENSE +0 -0
intentkit/__init__.py
CHANGED
intentkit/abstracts/skill.py
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
from abc import ABC, abstractmethod
|
|
2
|
-
from typing import Any, Dict,
|
|
2
|
+
from typing import Any, Dict, Optional
|
|
3
3
|
|
|
4
|
-
from intentkit.models.agent import Agent
|
|
4
|
+
from intentkit.models.agent import Agent
|
|
5
5
|
from intentkit.models.agent_data import AgentData, AgentQuota
|
|
6
6
|
|
|
7
7
|
|
|
@@ -58,60 +58,3 @@ class SkillStoreABC(ABC):
|
|
|
58
58
|
Agent quota if found, None otherwise
|
|
59
59
|
"""
|
|
60
60
|
pass
|
|
61
|
-
|
|
62
|
-
@staticmethod
|
|
63
|
-
@abstractmethod
|
|
64
|
-
async def list_autonomous_tasks(agent_id: str) -> List[AgentAutonomous]:
|
|
65
|
-
"""List all autonomous tasks for an agent.
|
|
66
|
-
|
|
67
|
-
Args:
|
|
68
|
-
agent_id: ID of the agent
|
|
69
|
-
|
|
70
|
-
Returns:
|
|
71
|
-
List[AgentAutonomous]: List of autonomous task configurations
|
|
72
|
-
"""
|
|
73
|
-
pass
|
|
74
|
-
|
|
75
|
-
@staticmethod
|
|
76
|
-
@abstractmethod
|
|
77
|
-
async def add_autonomous_task(
|
|
78
|
-
agent_id: str, task: AgentAutonomous
|
|
79
|
-
) -> AgentAutonomous:
|
|
80
|
-
"""Add a new autonomous task to an agent.
|
|
81
|
-
|
|
82
|
-
Args:
|
|
83
|
-
agent_id: ID of the agent
|
|
84
|
-
task: Autonomous task configuration
|
|
85
|
-
|
|
86
|
-
Returns:
|
|
87
|
-
AgentAutonomous: The created task
|
|
88
|
-
"""
|
|
89
|
-
pass
|
|
90
|
-
|
|
91
|
-
@staticmethod
|
|
92
|
-
@abstractmethod
|
|
93
|
-
async def delete_autonomous_task(agent_id: str, task_id: str) -> None:
|
|
94
|
-
"""Delete an autonomous task from an agent.
|
|
95
|
-
|
|
96
|
-
Args:
|
|
97
|
-
agent_id: ID of the agent
|
|
98
|
-
task_id: ID of the task to delete
|
|
99
|
-
"""
|
|
100
|
-
pass
|
|
101
|
-
|
|
102
|
-
@staticmethod
|
|
103
|
-
@abstractmethod
|
|
104
|
-
async def update_autonomous_task(
|
|
105
|
-
agent_id: str, task_id: str, task_updates: dict
|
|
106
|
-
) -> AgentAutonomous:
|
|
107
|
-
"""Update an autonomous task for an agent.
|
|
108
|
-
|
|
109
|
-
Args:
|
|
110
|
-
agent_id: ID of the agent
|
|
111
|
-
task_id: ID of the task to update
|
|
112
|
-
task_updates: Dictionary containing fields to update
|
|
113
|
-
|
|
114
|
-
Returns:
|
|
115
|
-
AgentAutonomous: The updated task
|
|
116
|
-
"""
|
|
117
|
-
pass
|
intentkit/clients/__init__.py
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
from intentkit.clients.cdp import
|
|
1
|
+
from intentkit.clients.cdp import get_cdp_client, get_evm_account, get_wallet_provider
|
|
2
2
|
from intentkit.clients.twitter import (
|
|
3
3
|
TwitterClient,
|
|
4
4
|
TwitterClientConfig,
|
|
@@ -10,7 +10,8 @@ __all__ = [
|
|
|
10
10
|
"TwitterClient",
|
|
11
11
|
"TwitterClientConfig",
|
|
12
12
|
"get_twitter_client",
|
|
13
|
-
"
|
|
13
|
+
"get_evm_account",
|
|
14
|
+
"get_cdp_client",
|
|
14
15
|
"get_wallet_provider",
|
|
15
16
|
"get_web3_client",
|
|
16
17
|
]
|
intentkit/clients/cdp.py
CHANGED
|
@@ -4,7 +4,7 @@ import logging
|
|
|
4
4
|
from typing import Dict, Optional, Tuple
|
|
5
5
|
|
|
6
6
|
from bip32 import BIP32
|
|
7
|
-
from cdp import CdpClient
|
|
7
|
+
from cdp import CdpClient, EvmServerAccount # noqa: E402
|
|
8
8
|
from coinbase_agentkit import ( # noqa: E402
|
|
9
9
|
CdpEvmWalletProvider,
|
|
10
10
|
CdpEvmWalletProviderConfig,
|
|
@@ -19,7 +19,7 @@ from intentkit.models.db import get_session
|
|
|
19
19
|
from intentkit.utils.error import IntentKitAPIError # noqa: E402
|
|
20
20
|
|
|
21
21
|
_wallet_providers: Dict[str, Tuple[str, str, CdpEvmWalletProvider]] = {}
|
|
22
|
-
|
|
22
|
+
_cdp_client: Optional[CdpClient] = None
|
|
23
23
|
|
|
24
24
|
logger = logging.getLogger(__name__)
|
|
25
25
|
|
|
@@ -59,25 +59,25 @@ def bip39_seed_to_eth_keys(seed_hex: str) -> Dict[str, str]:
|
|
|
59
59
|
}
|
|
60
60
|
|
|
61
61
|
|
|
62
|
-
def
|
|
63
|
-
global
|
|
64
|
-
if
|
|
65
|
-
return
|
|
62
|
+
def get_cdp_client() -> CdpClient:
|
|
63
|
+
global _cdp_client
|
|
64
|
+
if _cdp_client:
|
|
65
|
+
return _cdp_client
|
|
66
66
|
|
|
67
67
|
# Get credentials from global configuration
|
|
68
68
|
api_key_id = config.cdp_api_key_id
|
|
69
69
|
api_key_secret = config.cdp_api_key_secret
|
|
70
70
|
wallet_secret = config.cdp_wallet_secret
|
|
71
71
|
|
|
72
|
-
|
|
72
|
+
_cdp_client = CdpClient(
|
|
73
73
|
api_key_id=api_key_id,
|
|
74
74
|
api_key_secret=api_key_secret,
|
|
75
75
|
wallet_secret=wallet_secret,
|
|
76
76
|
)
|
|
77
|
-
return
|
|
77
|
+
return _cdp_client
|
|
78
78
|
|
|
79
79
|
|
|
80
|
-
|
|
80
|
+
def _assert_cdp_wallet_provider(agent: Agent) -> None:
|
|
81
81
|
if agent.wallet_provider != "cdp":
|
|
82
82
|
raise IntentKitAPIError(
|
|
83
83
|
400,
|
|
@@ -85,74 +85,93 @@ async def get_wallet_provider(agent: Agent) -> CdpEvmWalletProvider:
|
|
|
85
85
|
"Your agent wallet provider is not cdp but you selected a skill that requires a cdp wallet.",
|
|
86
86
|
)
|
|
87
87
|
|
|
88
|
-
if not agent.network_id:
|
|
89
|
-
raise IntentKitAPIError(
|
|
90
|
-
400,
|
|
91
|
-
"BadNetworkID",
|
|
92
|
-
"Your agent network ID is not set. Please set it in the agent config.",
|
|
93
|
-
)
|
|
94
88
|
|
|
95
|
-
|
|
89
|
+
async def _ensure_evm_account(
|
|
90
|
+
agent: Agent, agent_data: AgentData | None = None
|
|
91
|
+
) -> Tuple[EvmServerAccount, AgentData]:
|
|
92
|
+
cdp_client = get_cdp_client()
|
|
93
|
+
agent_data = agent_data or await AgentData.get(agent.id)
|
|
96
94
|
address = agent_data.evm_wallet_address
|
|
95
|
+
account: Optional[EvmServerAccount] = None
|
|
97
96
|
|
|
98
|
-
cache_entry = _wallet_providers.get(agent.id)
|
|
99
|
-
if cache_entry:
|
|
100
|
-
cached_network_id, cached_address, provider = cache_entry
|
|
101
|
-
if cached_network_id == agent.network_id:
|
|
102
|
-
if not address:
|
|
103
|
-
address = cached_address or provider.get_address()
|
|
104
|
-
if cached_address == address:
|
|
105
|
-
return provider
|
|
106
|
-
|
|
107
|
-
# Get credentials from global config
|
|
108
|
-
api_key_id = config.cdp_api_key_id
|
|
109
|
-
api_key_secret = config.cdp_api_key_secret
|
|
110
|
-
wallet_secret = config.cdp_wallet_secret
|
|
111
|
-
|
|
112
|
-
network_id = agent.network_id
|
|
113
|
-
|
|
114
|
-
# new agent or address not migrated yet
|
|
115
97
|
if not address:
|
|
116
|
-
cdp_client = get_origin_cdp_client()
|
|
117
|
-
# try migrating from v1 cdp_wallet_data
|
|
118
98
|
if agent_data.cdp_wallet_data:
|
|
119
99
|
wallet_data = json.loads(agent_data.cdp_wallet_data)
|
|
120
100
|
if not isinstance(wallet_data, dict):
|
|
121
101
|
raise ValueError("Invalid wallet data format")
|
|
122
102
|
if wallet_data.get("default_address_id") and wallet_data.get("seed"):
|
|
123
|
-
# verify seed and convert to pk
|
|
124
103
|
keys = bip39_seed_to_eth_keys(wallet_data["seed"])
|
|
125
104
|
if keys["address"] != wallet_data["default_address_id"]:
|
|
126
105
|
raise ValueError(
|
|
127
106
|
"Bad wallet data, seed does not match default_address_id"
|
|
128
107
|
)
|
|
129
|
-
# try to import wallet to v2
|
|
130
108
|
logger.info("Migrating wallet data to v2...")
|
|
131
|
-
await cdp_client.evm.import_account(
|
|
109
|
+
account = await cdp_client.evm.import_account(
|
|
132
110
|
name=agent.id,
|
|
133
111
|
private_key=keys["private_key"],
|
|
134
112
|
)
|
|
135
|
-
address =
|
|
113
|
+
address = account.address
|
|
136
114
|
logger.info("Migrated wallet data to v2 successfully: %s", address)
|
|
137
|
-
# still not address
|
|
138
115
|
if not address:
|
|
139
116
|
logger.info("Creating new wallet...")
|
|
140
|
-
|
|
117
|
+
account = await cdp_client.evm.create_account(
|
|
141
118
|
name=agent.id,
|
|
142
119
|
)
|
|
143
|
-
address =
|
|
120
|
+
address = account.address
|
|
144
121
|
logger.info("Created new wallet: %s", address)
|
|
145
122
|
|
|
146
123
|
agent_data.evm_wallet_address = address
|
|
147
124
|
await agent_data.save()
|
|
148
|
-
# Update agent slug with evm_wallet_address if slug is null or empty
|
|
149
125
|
if not agent.slug:
|
|
150
126
|
async with get_session() as db:
|
|
151
127
|
db_agent = await db.get(AgentTable, agent.id)
|
|
152
|
-
if db_agent:
|
|
128
|
+
if db_agent and not db_agent.slug:
|
|
153
129
|
db_agent.slug = agent_data.evm_wallet_address
|
|
154
130
|
await db.commit()
|
|
155
131
|
|
|
132
|
+
if account is None:
|
|
133
|
+
account = await cdp_client.evm.get_account(address=address)
|
|
134
|
+
|
|
135
|
+
return account, agent_data
|
|
136
|
+
|
|
137
|
+
|
|
138
|
+
async def get_evm_account(agent: Agent) -> EvmServerAccount:
|
|
139
|
+
_assert_cdp_wallet_provider(agent)
|
|
140
|
+
account, _ = await _ensure_evm_account(agent)
|
|
141
|
+
return account
|
|
142
|
+
|
|
143
|
+
|
|
144
|
+
async def get_wallet_provider(agent: Agent) -> CdpEvmWalletProvider:
|
|
145
|
+
_assert_cdp_wallet_provider(agent)
|
|
146
|
+
if not agent.network_id:
|
|
147
|
+
raise IntentKitAPIError(
|
|
148
|
+
400,
|
|
149
|
+
"BadNetworkID",
|
|
150
|
+
"Your agent network ID is not set. Please set it in the agent config.",
|
|
151
|
+
)
|
|
152
|
+
|
|
153
|
+
agent_data = await AgentData.get(agent.id)
|
|
154
|
+
address = agent_data.evm_wallet_address
|
|
155
|
+
|
|
156
|
+
cache_entry = _wallet_providers.get(agent.id)
|
|
157
|
+
if cache_entry:
|
|
158
|
+
cached_network_id, cached_address, provider = cache_entry
|
|
159
|
+
if cached_network_id == agent.network_id:
|
|
160
|
+
if not address:
|
|
161
|
+
address = cached_address or provider.get_address()
|
|
162
|
+
if cached_address == address:
|
|
163
|
+
return provider
|
|
164
|
+
|
|
165
|
+
account, agent_data = await _ensure_evm_account(agent, agent_data)
|
|
166
|
+
address = account.address
|
|
167
|
+
|
|
168
|
+
# Get credentials from global config
|
|
169
|
+
api_key_id = config.cdp_api_key_id
|
|
170
|
+
api_key_secret = config.cdp_api_key_secret
|
|
171
|
+
wallet_secret = config.cdp_wallet_secret
|
|
172
|
+
|
|
173
|
+
network_id = agent.network_id
|
|
174
|
+
|
|
156
175
|
wallet_provider_config = CdpEvmWalletProviderConfig(
|
|
157
176
|
api_key_id=api_key_id,
|
|
158
177
|
api_key_secret=api_key_secret,
|
intentkit/clients/twitter.py
CHANGED
|
@@ -11,7 +11,6 @@ from requests.auth import HTTPBasicAuth
|
|
|
11
11
|
from requests_oauthlib import OAuth2Session
|
|
12
12
|
from tweepy.asynchronous import AsyncClient
|
|
13
13
|
|
|
14
|
-
from intentkit.abstracts.skill import SkillStoreABC
|
|
15
14
|
from intentkit.abstracts.twitter import TwitterABC
|
|
16
15
|
from intentkit.models.agent_data import AgentData
|
|
17
16
|
from intentkit.models.redis import get_redis
|
|
@@ -80,33 +79,44 @@ class TwitterClient(TwitterABC):
|
|
|
80
79
|
|
|
81
80
|
Args:
|
|
82
81
|
agent_id: The ID of the agent
|
|
83
|
-
skill_store: The skill store for retrieving data
|
|
84
82
|
config: Configuration dictionary that may contain API keys
|
|
85
83
|
"""
|
|
86
84
|
|
|
87
|
-
def __init__(self, agent_id: str,
|
|
85
|
+
def __init__(self, agent_id: str, config: Dict) -> None:
|
|
88
86
|
"""Initialize the Twitter client.
|
|
89
87
|
|
|
90
88
|
Args:
|
|
91
89
|
agent_id: The ID of the agent
|
|
92
|
-
skill_store: The skill store for retrieving data
|
|
93
90
|
config: Configuration dictionary that may contain API keys
|
|
94
91
|
"""
|
|
95
92
|
self.agent_id = agent_id
|
|
96
93
|
self._client: Optional[AsyncClient] = None
|
|
97
|
-
self._skill_store = skill_store
|
|
98
94
|
self._agent_data: Optional[AgentData] = None
|
|
99
95
|
self.use_key = _is_self_key(config)
|
|
100
96
|
self._config = config
|
|
101
97
|
|
|
98
|
+
async def _get_agent_data(self) -> AgentData:
|
|
99
|
+
"""Retrieve cached agent data, loading from the database if needed."""
|
|
100
|
+
|
|
101
|
+
if not self._agent_data:
|
|
102
|
+
self._agent_data = await AgentData.get(self.agent_id)
|
|
103
|
+
return self._agent_data
|
|
104
|
+
|
|
105
|
+
async def _refresh_agent_data(self) -> AgentData:
|
|
106
|
+
"""Reload agent data from the database."""
|
|
107
|
+
|
|
108
|
+
self._agent_data = await AgentData.get(self.agent_id)
|
|
109
|
+
return self._agent_data
|
|
110
|
+
|
|
102
111
|
async def get_client(self) -> AsyncClient:
|
|
103
112
|
"""Get the initialized Twitter client.
|
|
104
113
|
|
|
105
114
|
Returns:
|
|
106
115
|
AsyncClient: The Twitter client if initialized
|
|
107
116
|
"""
|
|
108
|
-
|
|
109
|
-
|
|
117
|
+
|
|
118
|
+
agent_data = await self._get_agent_data()
|
|
119
|
+
|
|
110
120
|
if not self._client:
|
|
111
121
|
# Check if we have API keys in config
|
|
112
122
|
if self.use_key:
|
|
@@ -118,8 +128,8 @@ class TwitterClient(TwitterABC):
|
|
|
118
128
|
return_type=dict,
|
|
119
129
|
)
|
|
120
130
|
# refresh userinfo if needed
|
|
121
|
-
if not
|
|
122
|
-
|
|
131
|
+
if not agent_data.twitter_self_key_refreshed_at or (
|
|
132
|
+
agent_data.twitter_self_key_refreshed_at
|
|
123
133
|
< datetime.now(tz=timezone.utc) - timedelta(days=1)
|
|
124
134
|
):
|
|
125
135
|
me = await self._client.get_me(
|
|
@@ -127,7 +137,7 @@ class TwitterClient(TwitterABC):
|
|
|
127
137
|
user_fields="id,username,name,verified",
|
|
128
138
|
)
|
|
129
139
|
if me and "data" in me and "id" in me["data"]:
|
|
130
|
-
await
|
|
140
|
+
await AgentData.patch(
|
|
131
141
|
self.agent_id,
|
|
132
142
|
{
|
|
133
143
|
"twitter_id": me["data"]["id"],
|
|
@@ -139,9 +149,7 @@ class TwitterClient(TwitterABC):
|
|
|
139
149
|
),
|
|
140
150
|
},
|
|
141
151
|
)
|
|
142
|
-
|
|
143
|
-
self.agent_id
|
|
144
|
-
)
|
|
152
|
+
agent_data = await self._refresh_agent_data()
|
|
145
153
|
logger.info(
|
|
146
154
|
f"Twitter self key client initialized. "
|
|
147
155
|
f"Use API key: {self.use_key}, "
|
|
@@ -152,39 +160,40 @@ class TwitterClient(TwitterABC):
|
|
|
152
160
|
)
|
|
153
161
|
return self._client
|
|
154
162
|
# Otherwise try to get OAuth2 tokens from agent data
|
|
155
|
-
if not
|
|
163
|
+
if not agent_data.twitter_access_token:
|
|
156
164
|
raise Exception(f"[{self.agent_id}] Twitter access token not found")
|
|
157
|
-
if not
|
|
165
|
+
if not agent_data.twitter_access_token_expires_at:
|
|
158
166
|
raise Exception(
|
|
159
167
|
f"[{self.agent_id}] Twitter access token expiration not found"
|
|
160
168
|
)
|
|
161
|
-
if
|
|
169
|
+
if agent_data.twitter_access_token_expires_at <= datetime.now(
|
|
162
170
|
tz=timezone.utc
|
|
163
171
|
):
|
|
164
172
|
raise Exception(f"[{self.agent_id}] Twitter access token has expired")
|
|
165
173
|
self._client = AsyncClient(
|
|
166
|
-
bearer_token=
|
|
174
|
+
bearer_token=agent_data.twitter_access_token,
|
|
167
175
|
return_type=dict,
|
|
168
176
|
)
|
|
169
177
|
return self._client
|
|
178
|
+
|
|
170
179
|
if not self.use_key:
|
|
171
180
|
# check if access token has expired
|
|
172
|
-
if
|
|
181
|
+
if agent_data.twitter_access_token_expires_at <= datetime.now(
|
|
173
182
|
tz=timezone.utc
|
|
174
183
|
):
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
if self._agent_data.twitter_access_token_expires_at <= datetime.now(
|
|
184
|
+
agent_data = await self._refresh_agent_data()
|
|
185
|
+
if agent_data.twitter_access_token_expires_at <= datetime.now(
|
|
178
186
|
tz=timezone.utc
|
|
179
187
|
):
|
|
180
188
|
raise Exception(
|
|
181
189
|
f"[{self.agent_id}] Twitter access token has expired"
|
|
182
190
|
)
|
|
183
191
|
self._client = AsyncClient(
|
|
184
|
-
bearer_token=
|
|
192
|
+
bearer_token=agent_data.twitter_access_token,
|
|
185
193
|
return_type=dict,
|
|
186
194
|
)
|
|
187
195
|
return self._client
|
|
196
|
+
|
|
188
197
|
return self._client
|
|
189
198
|
|
|
190
199
|
@property
|
|
@@ -352,7 +361,7 @@ class TwitterClient(TwitterABC):
|
|
|
352
361
|
ValueError: If there's an error uploading the media.
|
|
353
362
|
"""
|
|
354
363
|
# Get agent data to access the token
|
|
355
|
-
agent_data = await
|
|
364
|
+
agent_data = await AgentData.get(agent_id)
|
|
356
365
|
if not agent_data.twitter_access_token:
|
|
357
366
|
raise ValueError("Only linked X account can post media")
|
|
358
367
|
|
|
@@ -423,15 +432,13 @@ def _is_self_key(config: Dict) -> bool:
|
|
|
423
432
|
return config.get("api_key_provider") == "agent_owner"
|
|
424
433
|
|
|
425
434
|
|
|
426
|
-
def get_twitter_client(
|
|
427
|
-
agent_id: str, skill_store: SkillStoreABC, config: Dict
|
|
428
|
-
) -> "TwitterClient":
|
|
435
|
+
def get_twitter_client(agent_id: str, config: Dict) -> "TwitterClient":
|
|
429
436
|
if _is_self_key(config):
|
|
430
437
|
if agent_id not in _clients_self_key:
|
|
431
|
-
_clients_self_key[agent_id] = TwitterClient(agent_id,
|
|
438
|
+
_clients_self_key[agent_id] = TwitterClient(agent_id, config)
|
|
432
439
|
return _clients_self_key[agent_id]
|
|
433
440
|
if agent_id not in _clients_linked:
|
|
434
|
-
_clients_linked[agent_id] = TwitterClient(agent_id,
|
|
441
|
+
_clients_linked[agent_id] = TwitterClient(agent_id, config)
|
|
435
442
|
return _clients_linked[agent_id]
|
|
436
443
|
|
|
437
444
|
|
intentkit/config/config.py
CHANGED
|
@@ -79,6 +79,7 @@ class Config:
|
|
|
79
79
|
self.debug_password = self.load("DEBUG_PASSWORD")
|
|
80
80
|
# Payment
|
|
81
81
|
self.payment_enabled = self.load("PAYMENT_ENABLED", "false") == "true"
|
|
82
|
+
self.x402_fee_address = self.load("X402_FEE_ADDRESS")
|
|
82
83
|
# Open API for agent
|
|
83
84
|
self.open_api_base_url = self.load("OPEN_API_BASE_URL", "http://localhost:8000")
|
|
84
85
|
# CDP - AgentKit 0.7.x Configuration
|