Files
browser-use/examples/apps/msg-use/scheduler.py
avocardio 00c23ad6ba Lints
2025-09-13 09:32:24 -07:00

287 lines
8.3 KiB
Python
Executable File

#!/usr/bin/env python3
"""
WhatsApp Message Scheduler - Send scheduled messages via WhatsApp Web
"""
import argparse
import asyncio
import json
import logging
import os
import random
import re
from datetime import datetime, timedelta
from pathlib import Path
def setup_environment(debug: bool):
if not debug:
os.environ['BROWSER_USE_SETUP_LOGGING'] = 'false'
os.environ['BROWSER_USE_LOGGING_LEVEL'] = 'critical'
logging.getLogger().setLevel(logging.CRITICAL)
else:
os.environ['BROWSER_USE_SETUP_LOGGING'] = 'true'
os.environ['BROWSER_USE_LOGGING_LEVEL'] = 'info'
parser = argparse.ArgumentParser(description='WhatsApp Scheduler - Send scheduled messages via WhatsApp Web')
parser.add_argument('--debug', action='store_true', help='Debug mode: show browser and verbose logs')
parser.add_argument('--test', action='store_true', help='Test mode: show what messages would be sent without sending them')
parser.add_argument('--auto', action='store_true', help='Auto mode: respond to unread messages every 30 minutes')
args = parser.parse_args()
setup_environment(args.debug)
from browser_use import Agent, BrowserSession
from browser_use.llm.google import ChatGoogle
GOOGLE_API_KEY = os.getenv('GOOGLE_API_KEY') or os.getenv('GEMINI_API_KEY')
USER_DATA_DIR = Path.home() / '.config' / 'whatsapp_scheduler' / 'browser_profile'
USER_DATA_DIR.mkdir(parents=True, exist_ok=True)
STORAGE_STATE_FILE = USER_DATA_DIR / 'storage_state.json'
async def parse_messages():
"""Parse messages.txt and extract scheduling info"""
messages_file = Path('messages.txt')
if not messages_file.exists():
print('❌ messages.txt not found!')
return []
import aiofiles
async with aiofiles.open(messages_file) as f:
content = await f.read()
llm = ChatGoogle(model='gemini-2.0-flash-exp', temperature=0.1, api_key=GOOGLE_API_KEY)
now = datetime.now()
prompt = f"""
Parse these WhatsApp message instructions and extract:
1. Contact name (extract just the name, not descriptions)
2. Message content (what to send)
3. Date and time (when to send)
Current date/time: {now.strftime('%Y-%m-%d %H:%M')}
Today is: {now.strftime('%Y-%m-%d')}
Current time is: {now.strftime('%H:%M')}
Instructions:
{content}
Return ONLY a JSON array with format:
[{{"contact": "name", "message": "text", "datetime": "YYYY-MM-DD HH:MM"}}]
CRITICAL: Transform instructions into actual messages:
QUOTED TEXT → Use exactly as-is:
- Text in "quotes" becomes the exact message
UNQUOTED INSTRUCTIONS → Generate actual content:
- If it's an instruction to write something → write the actual thing
- If it's an instruction to tell someone something → write what to tell them
- If it's an instruction to remind someone → write the actual reminder
- For multi-line content like poems: use single line with spacing, not line breaks
DO NOT copy the instruction - create the actual message content!
Time Rules:
- If only time given (like "at 15:30"), use TODAY
- If no date specified, assume TODAY
- If no year given, use current year
- Default time is 9:00 if not specified
- Extract names from parentheses: "hinge date (Camila)""Camila"
- "tomorrow" means {(now + timedelta(days=1)).strftime('%Y-%m-%d')}
- "next tuesday" or similar means the next occurrence of that day
"""
from browser_use.llm.messages import UserMessage
response = await llm.ainvoke([UserMessage(content=prompt)])
response_text = response.completion if hasattr(response, 'completion') else str(response)
# Extract JSON
json_match = re.search(r'\[.*?\]', response_text, re.DOTALL)
if json_match:
try:
messages = json.loads(json_match.group())
for msg in messages:
if 'message' in msg:
msg['message'] = re.sub(r'\n+', '', msg['message'])
msg['message'] = re.sub(r'\s+', ' ', msg['message']).strip()
return messages
except json.JSONDecodeError:
pass
return []
async def send_message(contact, message):
"""Send a WhatsApp message"""
print(f'\n📱 Sending to {contact}: {message}')
llm = ChatGoogle(model='gemini-2.0-flash-exp', temperature=0.3, api_key=GOOGLE_API_KEY)
task = f"""
Send WhatsApp message:
1. Go to https://web.whatsapp.com
2. Search for contact: {contact}
3. Click on the contact
4. Type message: {message}
5. Press Enter to send
6. Confirm sent
"""
browser = BrowserSession(
headless=not args.debug, # headless=False only when debug=True
user_data_dir=str(USER_DATA_DIR),
storage_state=str(STORAGE_STATE_FILE) if STORAGE_STATE_FILE.exists() else None,
)
agent = Agent(task=task, llm=llm, browser_session=browser)
await agent.run()
print(f'✅ Sent to {contact}')
async def auto_respond_to_unread():
"""Click unread tab and respond to messages"""
print('\nAuto-responding to unread messages...')
llm = ChatGoogle(model='gemini-2.0-flash-exp', temperature=0.3, api_key=GOOGLE_API_KEY)
task = """
1. Go to https://web.whatsapp.com
2. Wait for page to load
3. Click on the "Unread" filter tab
4. If there are unread messages:
- Click on each unread chat
- Read the last message
- Generate and send a friendly, contextual response
- Move to next unread chat
5. Report how many messages were responded to
"""
browser = BrowserSession(
headless=not args.debug,
user_data_dir=str(USER_DATA_DIR),
storage_state=str(STORAGE_STATE_FILE) if STORAGE_STATE_FILE.exists() else None,
)
agent = Agent(task=task, llm=llm, browser_session=browser)
result = await agent.run()
print('✅ Auto-response complete')
return result
async def main():
if not GOOGLE_API_KEY:
print('❌ Set GOOGLE_API_KEY or GEMINI_API_KEY environment variable')
return
print('WhatsApp Scheduler')
print(f'Profile: {USER_DATA_DIR}')
print()
# Auto mode - respond to unread messages periodically
if args.auto:
print('AUTO MODE - Responding to unread messages every ~30 minutes')
print('Press Ctrl+C to stop.\n')
while True:
try:
await auto_respond_to_unread()
# Wait 30 minutes +/- 5 minutes randomly
wait_minutes = 30 + random.randint(-5, 5)
print(f'\n⏰ Next check in {wait_minutes} minutes...')
await asyncio.sleep(wait_minutes * 60)
except KeyboardInterrupt:
print('\n\nAuto mode stopped by user')
break
except Exception as e:
print(f'\n❌ Error in auto mode: {e}')
print('Waiting 5 minutes before retry...')
await asyncio.sleep(300)
return
# Parse messages
print('Parsing messages.txt...')
messages = await parse_messages()
if not messages:
print('No messages found')
return
print(f'\nFound {len(messages)} messages:')
for msg in messages:
print(f'{msg["datetime"]}: {msg["message"][:30]}... to {msg["contact"]}')
now = datetime.now()
immediate = []
future = []
for msg in messages:
msg_time = datetime.strptime(msg['datetime'], '%Y-%m-%d %H:%M')
if msg_time <= now:
immediate.append(msg)
else:
future.append(msg)
if args.test:
print('\n=== TEST MODE - Preview ===')
if immediate:
print(f'\nWould send {len(immediate)} past-due messages NOW:')
for msg in immediate:
print(f' 📱 To {msg["contact"]}: {msg["message"]}')
if future:
print(f'\nWould monitor {len(future)} future messages:')
for msg in future:
print(f'{msg["datetime"]}: To {msg["contact"]}: {msg["message"]}')
print('\nTest mode complete. No messages sent.')
return
if immediate:
print(f'\nSending {len(immediate)} past-due messages NOW...')
for msg in immediate:
await send_message(msg['contact'], msg['message'])
if future:
print(f'\n⏰ Monitoring {len(future)} future messages...')
print('Press Ctrl+C to stop.\n')
last_status = None
while future:
now = datetime.now()
due = []
remaining = []
for msg in future:
msg_time = datetime.strptime(msg['datetime'], '%Y-%m-%d %H:%M')
if msg_time <= now:
due.append(msg)
else:
remaining.append(msg)
for msg in due:
print(f'\n⏰ Time reached for {msg["contact"]}')
await send_message(msg['contact'], msg['message'])
future = remaining
if future:
next_msg = min(future, key=lambda x: datetime.strptime(x['datetime'], '%Y-%m-%d %H:%M'))
current_status = f'Next: {next_msg["datetime"]} to {next_msg["contact"]}'
if current_status != last_status:
print(current_status)
last_status = current_status
await asyncio.sleep(30) # Check every 30 seconds
print('\n✅ All messages processed!')
if __name__ == '__main__':
asyncio.run(main())