Initial commit with translated description

This commit is contained in:
2026-03-29 10:19:32 +08:00
commit 54471bd63e
7 changed files with 1631 additions and 0 deletions

238
scripts/outlook-calendar.sh Normal file
View File

@@ -0,0 +1,238 @@
#!/bin/bash
# Outlook Calendar Operations
# Usage: outlook-calendar.sh <command> [args]
CONFIG_DIR="$HOME/.outlook-mcp"
CREDS_FILE="$CONFIG_DIR/credentials.json"
# Load token
ACCESS_TOKEN=$(jq -r '.access_token' "$CREDS_FILE" 2>/dev/null)
if [ -z "$ACCESS_TOKEN" ] || [ "$ACCESS_TOKEN" = "null" ]; then
echo "Error: No access token. Run setup first."
exit 1
fi
API="https://graph.microsoft.com/v1.0/me"
case "$1" in
events)
# List upcoming events
COUNT=${2:-10}
curl -s "$API/calendar/events?\$top=$COUNT&\$orderby=start/dateTime%20desc&\$select=id,subject,start,end,location,isAllDay" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Prefer: outlook.timezone=\"Europe/Madrid\"" | jq '.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, start: .value.start.dateTime[0:16], end: .value.end.dateTime[0:16], location: (.value.location.displayName // ""), id: .value.id[-20:]}'
;;
today)
# List today's events
TODAY_START=$(date -u +"%Y-%m-%dT00:00:00Z")
TODAY_END=$(date -u +"%Y-%m-%dT23:59:59Z")
curl -s "$API/calendarView?startDateTime=$TODAY_START&endDateTime=$TODAY_END&\$orderby=start/dateTime&\$select=id,subject,start,end,location" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Prefer: outlook.timezone=\"Europe/Madrid\"" | jq 'if .value then (.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, start: .value.start.dateTime[0:16], end: .value.end.dateTime[0:16], location: (.value.location.displayName // ""), id: .value.id[-20:]}) else {error: .error.message} end'
;;
week)
# List this week's events
WEEK_START=$(date -u +"%Y-%m-%dT00:00:00Z")
WEEK_END=$(date -u -d "+7 days" +"%Y-%m-%dT23:59:59Z" 2>/dev/null || date -u -v+7d +"%Y-%m-%dT23:59:59Z")
curl -s "$API/calendarView?startDateTime=$WEEK_START&endDateTime=$WEEK_END&\$orderby=start/dateTime&\$select=id,subject,start,end,location,isAllDay" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Prefer: outlook.timezone=\"Europe/Madrid\"" | jq 'if .value then (.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, start: .value.start.dateTime[0:16], end: .value.end.dateTime[0:16], location: (.value.location.displayName // ""), id: .value.id[-20:]}) else {error: .error.message} end'
;;
read)
# Read event details
EVENT_ID="$2"
FULL_ID=$(curl -s "$API/calendar/events?\$top=50&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$EVENT_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Event not found"
exit 1
fi
curl -s "$API/calendar/events/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Prefer: outlook.timezone=\"Europe/Madrid\"" | jq '{
subject,
start: .start.dateTime,
end: .end.dateTime,
location: .location.displayName,
body: (if .body.contentType == "html" then (.body.content | gsub("<[^>]*>"; "") | gsub("\\s+"; " ")[0:500]) else .body.content[0:500] end),
attendees: [.attendees[]?.emailAddress.address],
isOnline: .isOnlineMeeting,
link: .onlineMeeting.joinUrl
}'
;;
create)
# Create event: outlook-calendar.sh create "Subject" "2026-01-26T10:00" "2026-01-26T11:00" [location]
SUBJECT="$2"
START="$3"
END="$4"
LOCATION="${5:-}"
if [ -z "$SUBJECT" ] || [ -z "$START" ] || [ -z "$END" ]; then
echo "Usage: outlook-calendar.sh create <subject> <start> <end> [location]"
echo "Date format: YYYY-MM-DDTHH:MM (e.g., 2026-01-26T10:00)"
exit 1
fi
LOCATION_JSON=""
if [ -n "$LOCATION" ]; then
LOCATION_JSON=",\"location\": {\"displayName\": \"$LOCATION\"}"
fi
curl -s -X POST "$API/calendar/events" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{
\"subject\": \"$SUBJECT\",
\"start\": {\"dateTime\": \"$START\", \"timeZone\": \"Europe/Madrid\"},
\"end\": {\"dateTime\": \"$END\", \"timeZone\": \"Europe/Madrid\"}
$LOCATION_JSON
}" | jq '{status: "event created", subject: .subject, start: .start.dateTime[0:16], end: .end.dateTime[0:16], id: .id[-20:]}'
;;
quick)
# Quick event (1 hour from now or specified time)
SUBJECT="$2"
START_TIME="${3:-}"
if [ -z "$SUBJECT" ]; then
echo "Usage: outlook-calendar.sh quick <subject> [start-time]"
echo "If no time given, creates 1-hour event starting now"
exit 1
fi
if [ -z "$START_TIME" ]; then
START=$(date +"%Y-%m-%dT%H:%M")
END=$(date -d "+1 hour" +"%Y-%m-%dT%H:%M" 2>/dev/null || date -v+1H +"%Y-%m-%dT%H:%M")
else
START="$START_TIME"
# Parse and add 1 hour
END=$(date -d "$START_TIME + 1 hour" +"%Y-%m-%dT%H:%M" 2>/dev/null || echo "$START_TIME")
fi
curl -s -X POST "$API/calendar/events" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{
\"subject\": \"$SUBJECT\",
\"start\": {\"dateTime\": \"$START\", \"timeZone\": \"Europe/Madrid\"},
\"end\": {\"dateTime\": \"$END\", \"timeZone\": \"Europe/Madrid\"}
}" | jq '{status: "quick event created", subject: .subject, start: .start.dateTime[0:16], end: .end.dateTime[0:16], id: .id[-20:]}'
;;
delete)
# Delete event
EVENT_ID="$2"
FULL_ID=$(curl -s "$API/calendar/events?\$top=50&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$EVENT_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Event not found"
exit 1
fi
RESULT=$(curl -s -w "\n%{http_code}" -X DELETE "$API/calendar/events/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN")
HTTP_CODE=$(echo "$RESULT" | tail -1)
if [ "$HTTP_CODE" = "204" ]; then
echo "{\"status\": \"event deleted\", \"id\": \"$EVENT_ID\"}"
else
echo "$RESULT" | head -n -1 | jq '.error // .'
fi
;;
update)
# Update event: outlook-calendar.sh update <id> <field> <value>
EVENT_ID="$2"
FIELD="$3"
VALUE="$4"
if [ -z "$FIELD" ] || [ -z "$VALUE" ]; then
echo "Usage: outlook-calendar.sh update <id> <field> <value>"
echo "Fields: subject, location, start, end"
exit 1
fi
FULL_ID=$(curl -s "$API/calendar/events?\$top=50&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$EVENT_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Event not found"
exit 1
fi
case "$FIELD" in
subject)
BODY="{\"subject\": \"$VALUE\"}"
;;
location)
BODY="{\"location\": {\"displayName\": \"$VALUE\"}}"
;;
start)
BODY="{\"start\": {\"dateTime\": \"$VALUE\", \"timeZone\": \"Europe/Madrid\"}}"
;;
end)
BODY="{\"end\": {\"dateTime\": \"$VALUE\", \"timeZone\": \"Europe/Madrid\"}}"
;;
*)
echo "Unknown field: $FIELD"
exit 1
;;
esac
curl -s -X PATCH "$API/calendar/events/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "$BODY" | jq '{status: "event updated", subject: .subject, start: .start.dateTime[0:16], end: .end.dateTime[0:16], id: .id[-20:]}'
;;
calendars)
# List all calendars
curl -s "$API/calendars" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '.value[] | {name: .name, color: .color, canEdit: .canEdit, id: .id[-20:]}'
;;
free)
# Check free/busy for a time range
START="$2"
END="$3"
if [ -z "$START" ] || [ -z "$END" ]; then
echo "Usage: outlook-calendar.sh free <start> <end>"
echo "Date format: YYYY-MM-DDTHH:MM"
exit 1
fi
curl -s "$API/calendarView?startDateTime=${START}:00Z&endDateTime=${END}:00Z&\$select=subject,start,end" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq 'if (.value | length) == 0 then {status: "free", start: "'"$START"'", end: "'"$END"'"} else {status: "busy", events: [.value[].subject]} end'
;;
*)
echo "Usage: outlook-calendar.sh <command> [args]"
echo ""
echo "VIEW:"
echo " events [count] - List upcoming events"
echo " today - Today's events"
echo " week - This week's events"
echo " read <id> - Event details"
echo " calendars - List all calendars"
echo " free <start> <end> - Check availability"
echo ""
echo "CREATE:"
echo " create <subj> <start> <end> [loc] - Create event"
echo " quick <subject> [time] - Quick 1-hour event"
echo ""
echo "MANAGE:"
echo " update <id> <field> <val> - Update event"
echo " delete <id> - Delete event"
echo ""
echo "Date format: YYYY-MM-DDTHH:MM (e.g., 2026-01-26T10:00)"
;;
esac

707
scripts/outlook-mail.sh Normal file
View File

@@ -0,0 +1,707 @@
#!/bin/bash
# Outlook Mail Operations
# Usage: outlook-mail.sh <command> [args]
CONFIG_DIR="$HOME/.outlook-mcp"
CREDS_FILE="$CONFIG_DIR/credentials.json"
# Load token
ACCESS_TOKEN=$(jq -r '.access_token' "$CREDS_FILE" 2>/dev/null)
if [ -z "$ACCESS_TOKEN" ] || [ "$ACCESS_TOKEN" = "null" ]; then
echo "Error: No access token. Run setup first."
exit 1
fi
API="https://graph.microsoft.com/v1.0/me"
case "$1" in
inbox)
# List inbox messages
COUNT=${2:-10}
curl -s "$API/messages?\$top=$COUNT&\$orderby=receivedDateTime%20desc&\$select=id,subject,from,receivedDateTime,isRead" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, from: .value.from.emailAddress.address, date: .value.receivedDateTime[0:16], read: .value.isRead, id: .value.id[-20:]}'
;;
unread)
# List unread messages
COUNT=${2:-20}
curl -s "$API/messages?\$filter=isRead%20eq%20false&\$top=$COUNT&\$orderby=receivedDateTime%20desc&\$select=id,subject,from,receivedDateTime" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, from: .value.from.emailAddress.address, date: .value.receivedDateTime[0:16], id: .value.id[-20:]}'
;;
search)
# Search emails
QUERY="$2"
COUNT=${3:-20}
curl -s "$API/messages?\$search=\"$QUERY\"&\$top=$COUNT&\$select=id,subject,from,receivedDateTime" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, from: .value.from.emailAddress.address, date: .value.receivedDateTime[0:16], id: .value.id[-20:]}'
;;
read)
# Read specific email by ID (partial ID match - uses last 20 chars)
MSG_ID="$2"
# First find full ID (search by suffix)
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found. Use the ID shown in inbox/unread/search results."
exit 1
fi
# Get message and extract text from HTML body
curl -s "$API/messages/$FULL_ID?\$select=subject,from,receivedDateTime,body,toRecipients" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '{
subject,
from: .from.emailAddress,
to: [.toRecipients[].emailAddress.address],
date: .receivedDateTime,
body: (if .body.contentType == "html" then (.body.content | gsub("<[^>]*>"; "") | gsub("\\s+"; " ") | gsub("&nbsp;"; " ") | .[0:2000]) else .body.content[0:2000] end)
}'
;;
mark-read)
# Mark message as read
MSG_ID="$2"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
curl -s -X PATCH "$API/messages/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d '{"isRead": true}' | jq '{status: "marked as read", subject: .subject, id: .id[-20:]}'
;;
folders)
# List mail folders
curl -s "$API/mailFolders" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '.value[] | {name: .displayName, total: .totalItemCount, unread: .unreadItemCount}'
;;
stats)
# Get inbox stats
curl -s "$API/mailFolders/inbox" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '{folder: .displayName, total: .totalItemCount, unread: .unreadItemCount}'
;;
send)
# Send email: outlook-mail.sh send "to@email.com" "Subject" "Body"
TO="$2"
SUBJECT="$3"
BODY="$4"
if [ -z "$TO" ] || [ -z "$SUBJECT" ]; then
echo "Usage: outlook-mail.sh send <to> <subject> <body>"
exit 1
fi
RESULT=$(curl -s -w "\n%{http_code}" -X POST "$API/sendMail" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{
\"message\": {
\"subject\": \"$SUBJECT\",
\"body\": {\"contentType\": \"Text\", \"content\": \"$BODY\"},
\"toRecipients\": [{\"emailAddress\": {\"address\": \"$TO\"}}]
}
}")
HTTP_CODE=$(echo "$RESULT" | tail -1)
if [ "$HTTP_CODE" = "202" ]; then
echo "{\"status\": \"sent\", \"to\": \"$TO\", \"subject\": \"$SUBJECT\"}"
else
echo "$RESULT" | head -n -1 | jq '.error // .'
fi
;;
mark-unread)
# Mark message as unread
MSG_ID="$2"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
curl -s -X PATCH "$API/messages/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d '{"isRead": false}' | jq '{status: "marked as unread", subject: .subject, id: .id[-20:]}'
;;
delete)
# Move message to trash
MSG_ID="$2"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
curl -s -X POST "$API/messages/$FULL_ID/move" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d '{"destinationId": "deleteditems"}' | jq '{status: "moved to trash", subject: .subject, id: .id[-20:]}'
;;
archive)
# Move message to archive
MSG_ID="$2"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
curl -s -X POST "$API/messages/$FULL_ID/move" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d '{"destinationId": "archive"}' | jq '{status: "archived", subject: .subject, id: .id[-20:]}'
;;
flag)
# Flag message as important
MSG_ID="$2"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
curl -s -X PATCH "$API/messages/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d '{"flag": {"flagStatus": "flagged"}}' | jq '{status: "flagged", subject: .subject, id: .id[-20:]}'
;;
unflag)
# Remove flag from message
MSG_ID="$2"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
curl -s -X PATCH "$API/messages/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d '{"flag": {"flagStatus": "notFlagged"}}' | jq '{status: "unflagged", subject: .subject, id: .id[-20:]}'
;;
from)
# List emails from specific sender (uses search - more reliable than filter)
SENDER="$2"
COUNT=${3:-20}
curl -s "$API/messages?\$search=\"from:$SENDER\"&\$top=$COUNT&\$select=id,subject,from,receivedDateTime,isRead" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq 'if .value then (.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, from: .value.from.emailAddress.address, date: .value.receivedDateTime[0:16], read: .value.isRead, id: .value.id[-20:]}) else {error: .error.message} end'
;;
attachments)
# List attachments for a message
MSG_ID="$2"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
curl -s "$API/messages/$FULL_ID/attachments" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '.value[] | {name: .name, size: .size, contentType: .contentType, id: .id}'
;;
reply)
# Reply to a message: outlook-mail.sh reply <id> "Reply body"
MSG_ID="$2"
BODY="$3"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
RESULT=$(curl -s -w "\n%{http_code}" -X POST "$API/messages/$FULL_ID/reply" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{\"comment\": \"$BODY\"}")
HTTP_CODE=$(echo "$RESULT" | tail -1)
if [ "$HTTP_CODE" = "202" ]; then
echo "{\"status\": \"reply sent\", \"id\": \"$MSG_ID\"}"
else
echo "$RESULT" | head -n -1 | jq '.error // .'
fi
;;
move)
# Move message to folder: outlook-mail.sh move <id> <folder>
MSG_ID="$2"
FOLDER="$3"
if [ -z "$FOLDER" ]; then
echo "Usage: outlook-mail.sh move <id> <folder>"
echo "Use 'folders' command to see available folders"
exit 1
fi
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
# Get folder ID by name (case-insensitive)
FOLDER_LOWER=$(echo "$FOLDER" | tr '[:upper:]' '[:lower:]')
FOLDER_ID=$(curl -s "$API/mailFolders" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select((.displayName | ascii_downcase) == \"$FOLDER_LOWER\") | .id" | head -1)
if [ -z "$FOLDER_ID" ]; then
echo "Folder not found: $FOLDER"
echo "Available folders:"
curl -s "$API/mailFolders" -H "Authorization: Bearer $ACCESS_TOKEN" | jq -r '.value[].displayName'
exit 1
fi
curl -s -X POST "$API/messages/$FULL_ID/move" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{\"destinationId\": \"$FOLDER_ID\"}" | jq '{status: "moved", folder: "'"$FOLDER"'", subject: .subject, id: .id[-20:]}'
;;
draft)
# Create a draft email (not sent)
TO="$2"
SUBJECT="$3"
BODY="$4"
if [ -z "$TO" ] || [ -z "$SUBJECT" ]; then
echo "Usage: outlook-mail.sh draft <to> <subject> <body>"
exit 1
fi
curl -s -X POST "$API/messages" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{
\"subject\": \"$SUBJECT\",
\"body\": {\"contentType\": \"Text\", \"content\": \"$BODY\"},
\"toRecipients\": [{\"emailAddress\": {\"address\": \"$TO\"}}]
}" | jq '{status: "draft created", subject: .subject, to: .toRecipients[0].emailAddress.address, id: .id[-20:]}'
;;
drafts)
# List draft emails
COUNT=${2:-10}
curl -s "$API/mailFolders/drafts/messages?\$top=$COUNT&\$select=id,subject,toRecipients,createdDateTime" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, to: (.value.toRecipients[0].emailAddress.address // "no recipient"), date: .value.createdDateTime[0:16], id: .value.id[-20:]}'
;;
send-draft)
# Send an existing draft
MSG_ID="$2"
# Search in drafts folder
FULL_ID=$(curl -s "$API/mailFolders/drafts/messages?\$top=50&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Draft not found"
exit 1
fi
RESULT=$(curl -s -w "\n%{http_code}" -X POST "$API/messages/$FULL_ID/send" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Length: 0")
HTTP_CODE=$(echo "$RESULT" | tail -1)
if [ "$HTTP_CODE" = "202" ]; then
echo "{\"status\": \"draft sent\", \"id\": \"$MSG_ID\"}"
else
echo "$RESULT" | head -n -1 | jq '.error // .'
fi
;;
forward)
# Forward an email: outlook-mail.sh forward <id> <to> [comment]
MSG_ID="$2"
TO="$3"
COMMENT="${4:-}"
if [ -z "$TO" ]; then
echo "Usage: outlook-mail.sh forward <id> <to> [comment]"
exit 1
fi
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
RESULT=$(curl -s -w "\n%{http_code}" -X POST "$API/messages/$FULL_ID/forward" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{
\"comment\": \"$COMMENT\",
\"toRecipients\": [{\"emailAddress\": {\"address\": \"$TO\"}}]
}")
HTTP_CODE=$(echo "$RESULT" | tail -1)
if [ "$HTTP_CODE" = "202" ]; then
echo "{\"status\": \"forwarded\", \"to\": \"$TO\", \"id\": \"$MSG_ID\"}"
else
echo "$RESULT" | head -n -1 | jq '.error // .'
fi
;;
download)
# Download an attachment: outlook-mail.sh download <msg-id> <attachment-name> [output-path]
MSG_ID="$2"
ATT_NAME="$3"
OUTPUT="${4:-.}"
if [ -z "$ATT_NAME" ]; then
echo "Usage: outlook-mail.sh download <msg-id> <attachment-name> [output-path]"
echo "Use 'attachments <id>' to see available attachments"
exit 1
fi
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
# Get attachment by name
ATT_DATA=$(curl -s "$API/messages/$FULL_ID/attachments" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.name == \"$ATT_NAME\")")
if [ -z "$ATT_DATA" ]; then
echo "Attachment not found: $ATT_NAME"
echo "Available attachments:"
curl -s "$API/messages/$FULL_ID/attachments" -H "Authorization: Bearer $ACCESS_TOKEN" | jq -r '.value[].name'
exit 1
fi
# Get content and decode
ATT_ID=$(echo "$ATT_DATA" | jq -r '.id')
CONTENT=$(curl -s "$API/messages/$FULL_ID/attachments/$ATT_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r '.contentBytes')
OUTPUT_FILE="$OUTPUT/$ATT_NAME"
echo "$CONTENT" | base64 -d > "$OUTPUT_FILE"
if [ -f "$OUTPUT_FILE" ]; then
SIZE=$(stat -c%s "$OUTPUT_FILE" 2>/dev/null || stat -f%z "$OUTPUT_FILE")
echo "{\"status\": \"downloaded\", \"file\": \"$OUTPUT_FILE\", \"size\": $SIZE}"
else
echo "{\"error\": \"Failed to save file\"}"
exit 1
fi
;;
create-folder)
# Create a new mail folder
FOLDER_NAME="$2"
PARENT="${3:-}"
if [ -z "$FOLDER_NAME" ]; then
echo "Usage: outlook-mail.sh create-folder <name> [parent-folder]"
exit 1
fi
if [ -n "$PARENT" ]; then
# Get parent folder ID
PARENT_LOWER=$(echo "$PARENT" | tr '[:upper:]' '[:lower:]')
PARENT_ID=$(curl -s "$API/mailFolders" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select((.displayName | ascii_downcase) == \"$PARENT_LOWER\") | .id" | head -1)
if [ -z "$PARENT_ID" ]; then
echo "Parent folder not found: $PARENT"
exit 1
fi
curl -s -X POST "$API/mailFolders/$PARENT_ID/childFolders" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{\"displayName\": \"$FOLDER_NAME\"}" | jq '{status: "folder created", name: .displayName, parent: "'"$PARENT"'", id: .id[-20:]}'
else
curl -s -X POST "$API/mailFolders" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{\"displayName\": \"$FOLDER_NAME\"}" | jq '{status: "folder created", name: .displayName, id: .id[-20:]}'
fi
;;
delete-folder)
# Delete a mail folder
FOLDER_NAME="$2"
if [ -z "$FOLDER_NAME" ]; then
echo "Usage: outlook-mail.sh delete-folder <name>"
exit 1
fi
FOLDER_LOWER=$(echo "$FOLDER_NAME" | tr '[:upper:]' '[:lower:]')
FOLDER_ID=$(curl -s "$API/mailFolders" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select((.displayName | ascii_downcase) == \"$FOLDER_LOWER\") | .id" | head -1)
if [ -z "$FOLDER_ID" ]; then
echo "Folder not found: $FOLDER_NAME"
exit 1
fi
RESULT=$(curl -s -w "\n%{http_code}" -X DELETE "$API/mailFolders/$FOLDER_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN")
HTTP_CODE=$(echo "$RESULT" | tail -1)
if [ "$HTTP_CODE" = "204" ]; then
echo "{\"status\": \"folder deleted\", \"name\": \"$FOLDER_NAME\"}"
else
echo "$RESULT" | head -n -1 | jq '.error // .'
fi
;;
bulk-read)
# Mark multiple messages as read: outlook-mail.sh bulk-read <id1> <id2> ...
shift
if [ $# -eq 0 ]; then
echo "Usage: outlook-mail.sh bulk-read <id1> <id2> <id3> ..."
exit 1
fi
SUCCESS=0
FAILED=0
for MSG_ID in "$@"; do
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -n "$FULL_ID" ]; then
curl -s -X PATCH "$API/messages/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d '{"isRead": true}' > /dev/null
SUCCESS=$((SUCCESS + 1))
else
FAILED=$((FAILED + 1))
fi
done
echo "{\"status\": \"bulk operation complete\", \"marked_read\": $SUCCESS, \"not_found\": $FAILED}"
;;
bulk-delete)
# Delete multiple messages: outlook-mail.sh bulk-delete <id1> <id2> ...
shift
if [ $# -eq 0 ]; then
echo "Usage: outlook-mail.sh bulk-delete <id1> <id2> <id3> ..."
exit 1
fi
SUCCESS=0
FAILED=0
for MSG_ID in "$@"; do
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -n "$FULL_ID" ]; then
curl -s -X POST "$API/messages/$FULL_ID/move" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d '{"destinationId": "deleteditems"}' > /dev/null
SUCCESS=$((SUCCESS + 1))
else
FAILED=$((FAILED + 1))
fi
done
echo "{\"status\": \"bulk delete complete\", \"deleted\": $SUCCESS, \"not_found\": $FAILED}"
;;
categories)
# List available categories (like Gmail labels)
curl -s "https://graph.microsoft.com/v1.0/me/outlook/masterCategories" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '.value[] | {name: .displayName, color: .color, id: .id[0:8]}'
;;
categorize)
# Add category to message: outlook-mail.sh categorize <id> <category-name>
MSG_ID="$2"
CATEGORY="$3"
if [ -z "$CATEGORY" ]; then
echo "Usage: outlook-mail.sh categorize <id> <category-name>"
echo "Use 'categories' to see available categories"
exit 1
fi
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id,categories" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
# Get current categories and add new one
CURRENT=$(curl -s "$API/messages/$FULL_ID?\$select=categories" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r '.categories | join(",")')
if [ -n "$CURRENT" ]; then
NEW_CATS="[\"$CURRENT\",\"$CATEGORY\"]"
else
NEW_CATS="[\"$CATEGORY\"]"
fi
curl -s -X PATCH "$API/messages/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d "{\"categories\": $NEW_CATS}" | jq '{status: "categorized", subject: .subject, categories: .categories, id: .id[-20:]}'
;;
uncategorize)
# Remove all categories from message
MSG_ID="$2"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
curl -s -X PATCH "$API/messages/$FULL_ID" \
-H "Authorization: Bearer $ACCESS_TOKEN" \
-H "Content-Type: application/json" \
-d '{"categories": []}' | jq '{status: "categories removed", subject: .subject, id: .id[-20:]}'
;;
focused)
# List focused inbox (important emails)
COUNT=${2:-10}
curl -s "$API/messages?\$filter=inferenceClassification%20eq%20'focused'&\$top=$COUNT&\$orderby=receivedDateTime%20desc&\$select=id,subject,from,receivedDateTime" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq 'if .value then (.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, from: .value.from.emailAddress.address, date: .value.receivedDateTime[0:16], id: .value.id[-20:]}) else {info: "Focused inbox not available or empty"} end'
;;
other)
# List "other" inbox (non-focused emails)
COUNT=${2:-10}
curl -s "$API/messages?\$filter=inferenceClassification%20eq%20'other'&\$top=$COUNT&\$orderby=receivedDateTime%20desc&\$select=id,subject,from,receivedDateTime" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq 'if .value then (.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, from: .value.from.emailAddress.address, date: .value.receivedDateTime[0:16], id: .value.id[-20:]}) else {info: "Other inbox not available or empty"} end'
;;
thread)
# List emails in same conversation/thread (by subject keyword)
MSG_ID="$2"
FULL_ID=$(curl -s "$API/messages?\$top=100&\$select=id" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r ".value[] | select(.id | endswith(\"$MSG_ID\")) | .id" | head -1)
if [ -z "$FULL_ID" ]; then
echo "Message not found"
exit 1
fi
# Get subject, clean prefixes
SUBJECT=$(curl -s "$API/messages/$FULL_ID?\$select=subject" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq -r '.subject' | sed 's/^RE: //i' | sed 's/^FW: //i' | sed 's/^Fwd: //i')
# Get longest word as keyword (more specific)
KEYWORD=$(echo "$SUBJECT" | tr ' ' '\n' | awk '{print length, $0}' | sort -rn | head -1 | cut -d' ' -f2)
if [ -z "$KEYWORD" ] || [ ${#KEYWORD} -lt 4 ]; then
KEYWORD=$(echo "$SUBJECT" | cut -d' ' -f1)
fi
echo "Searching thread by keyword: $KEYWORD"
# Search by single keyword
curl -s "$API/messages?\$search=\"$KEYWORD\"&\$top=20&\$select=id,subject,from,receivedDateTime" \
-H "Authorization: Bearer $ACCESS_TOKEN" | jq '.value | to_entries | .[] | {n: (.key + 1), subject: .value.subject, from: .value.from.emailAddress.address, date: .value.receivedDateTime[0:16], id: .value.id[-20:]}'
;;
*)
echo "Usage: outlook-mail.sh <command> [args]"
echo ""
echo "READING:"
echo " inbox [count] - List latest emails (default: 10)"
echo " unread [count] - List unread emails"
echo " focused [count] - Focused/important inbox"
echo " other [count] - Other/low-priority inbox"
echo " search \"query\" [count] - Search emails"
echo " from <email> [count] - List emails from sender"
echo " read <id> - Read email content"
echo " thread <id> - View conversation thread"
echo " attachments <id> - List email attachments"
echo ""
echo "MANAGING:"
echo " mark-read <id> - Mark as read"
echo " mark-unread <id> - Mark as unread"
echo " flag <id> - Flag as important"
echo " unflag <id> - Remove flag"
echo " delete <id> - Move to trash"
echo " archive <id> - Move to archive"
echo " move <id> <folder> - Move to folder"
echo ""
echo "CATEGORIES (like Gmail labels):"
echo " categories - List available categories"
echo " categorize <id> <name> - Add category to email"
echo " uncategorize <id> - Remove categories"
echo ""
echo "SENDING:"
echo " send <to> <subj> <body> - Send new email"
echo " reply <id> \"body\" - Reply to email"
echo " forward <id> <to> [msg] - Forward email"
echo ""
echo "DRAFTS:"
echo " draft <to> <subj> <body> - Create draft (not sent)"
echo " drafts [count] - List drafts"
echo " send-draft <id> - Send a draft"
echo ""
echo "ATTACHMENTS:"
echo " attachments <id> - List attachments"
echo " download <id> <name> [path] - Download attachment"
echo ""
echo "FOLDERS:"
echo " folders - List mail folders"
echo " create-folder <name> [parent] - Create folder"
echo " delete-folder <name> - Delete folder"
echo ""
echo "BULK OPERATIONS:"
echo " bulk-read <id1> <id2>... - Mark multiple as read"
echo " bulk-delete <id1> <id2>... - Delete multiple"
echo ""
echo "INFO:"
echo " stats - Inbox statistics"
;;
esac

251
scripts/outlook-setup.sh Normal file
View File

@@ -0,0 +1,251 @@
#!/bin/bash
# Outlook OAuth Setup via Azure CLI
# Automatically creates App Registration and configures OAuth
set -e
CONFIG_DIR="$HOME/.outlook-mcp"
CONFIG_FILE="$CONFIG_DIR/config.json"
CREDS_FILE="$CONFIG_DIR/credentials.json"
APP_NAME="Clawdbot-Outlook"
REDIRECT_URI="http://localhost"
SCOPES="https://graph.microsoft.com/Mail.ReadWrite https://graph.microsoft.com/Mail.Send https://graph.microsoft.com/Calendars.ReadWrite offline_access"
# Colors
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
NC='\033[0m' # No Color
echo -e "${BLUE}=== Outlook OAuth Setup ===${NC}"
echo ""
# Check prerequisites
check_prereqs() {
if ! command -v az &> /dev/null; then
echo -e "${RED}Error: Azure CLI not installed${NC}"
echo "Install with: curl -sL https://aka.ms/InstallAzureCLIDeb | sudo bash"
exit 1
fi
if ! command -v jq &> /dev/null; then
echo -e "${RED}Error: jq not installed${NC}"
echo "Install with: sudo apt install jq"
exit 1
fi
}
# Step 1: Azure Login
azure_login() {
echo -e "${YELLOW}Step 1: Azure Login${NC}"
# Check if already logged in
if az account show &> /dev/null; then
CURRENT_USER=$(az account show --query user.name -o tsv)
echo -e "Currently logged in as: ${GREEN}$CURRENT_USER${NC}"
read -p "Continue with this account? [Y/n] " -n 1 -r
echo
if [[ $REPLY =~ ^[Nn]$ ]]; then
az logout 2>/dev/null || true
else
return 0
fi
fi
echo "Opening browser for Azure login..."
echo "(If no browser available, use: az login --use-device-code)"
if ! az login --use-device-code; then
echo -e "${RED}Login failed${NC}"
exit 1
fi
echo -e "${GREEN}✓ Logged in successfully${NC}"
}
# Step 2: Create App Registration
create_app() {
echo ""
echo -e "${YELLOW}Step 2: Creating App Registration${NC}"
# Check if app already exists
EXISTING_APP=$(az ad app list --display-name "$APP_NAME" --query "[0].appId" -o tsv 2>/dev/null)
if [ -n "$EXISTING_APP" ] && [ "$EXISTING_APP" != "null" ]; then
echo -e "App '$APP_NAME' already exists: ${BLUE}$EXISTING_APP${NC}"
read -p "Use existing app? [Y/n] " -n 1 -r
echo
if [[ ! $REPLY =~ ^[Nn]$ ]]; then
APP_ID="$EXISTING_APP"
echo -e "${GREEN}✓ Using existing app${NC}"
return 0
fi
APP_NAME="$APP_NAME-$(date +%s)"
echo "Creating new app: $APP_NAME"
fi
# Create new app
APP_RESULT=$(az ad app create \
--display-name "$APP_NAME" \
--sign-in-audience "AzureADandPersonalMicrosoftAccount" \
--web-redirect-uris "$REDIRECT_URI" \
--query "{appId: appId, objectId: id}" -o json)
APP_ID=$(echo "$APP_RESULT" | jq -r '.appId')
echo -e "${GREEN}✓ App created: $APP_ID${NC}"
}
# Step 3: Create Client Secret
create_secret() {
echo ""
echo -e "${YELLOW}Step 3: Creating Client Secret${NC}"
SECRET_RESULT=$(az ad app credential reset \
--id "$APP_ID" \
--display-name "clawdbot-secret" \
--years 2 \
--query "{clientId: appId, clientSecret: password}" -o json 2>&1)
CLIENT_ID=$(echo "$SECRET_RESULT" | jq -r '.clientId')
CLIENT_SECRET=$(echo "$SECRET_RESULT" | jq -r '.clientSecret')
if [ -z "$CLIENT_SECRET" ] || [ "$CLIENT_SECRET" = "null" ]; then
echo -e "${RED}Failed to create secret${NC}"
exit 1
fi
echo -e "${GREEN}✓ Secret created${NC}"
}
# Step 4: Add API Permissions
add_permissions() {
echo ""
echo -e "${YELLOW}Step 4: Adding API Permissions${NC}"
# Microsoft Graph API ID
GRAPH_API="00000003-0000-0000-c000-000000000000"
# Get permission IDs
MAIL_RW_ID=$(az ad sp show --id "$GRAPH_API" --query "oauth2PermissionScopes[?value=='Mail.ReadWrite'].id" -o tsv 2>/dev/null)
MAIL_SEND_ID=$(az ad sp show --id "$GRAPH_API" --query "oauth2PermissionScopes[?value=='Mail.Send'].id" -o tsv 2>/dev/null)
CAL_RW_ID=$(az ad sp show --id "$GRAPH_API" --query "oauth2PermissionScopes[?value=='Calendars.ReadWrite'].id" -o tsv 2>/dev/null)
USER_READ_ID=$(az ad sp show --id "$GRAPH_API" --query "oauth2PermissionScopes[?value=='User.Read'].id" -o tsv 2>/dev/null)
# Add permissions
az ad app permission add --id "$APP_ID" \
--api "$GRAPH_API" \
--api-permissions "$MAIL_RW_ID=Scope" "$MAIL_SEND_ID=Scope" "$CAL_RW_ID=Scope" "$USER_READ_ID=Scope" 2>/dev/null || true
echo -e "${GREEN}✓ Permissions added (Mail.ReadWrite, Mail.Send, Calendars.ReadWrite, User.Read)${NC}"
}
# Step 5: Save Config
save_config() {
echo ""
echo -e "${YELLOW}Step 5: Saving Configuration${NC}"
mkdir -p "$CONFIG_DIR"
cat > "$CONFIG_FILE" << EOF
{
"client_id": "$CLIENT_ID",
"client_secret": "$CLIENT_SECRET"
}
EOF
chmod 600 "$CONFIG_FILE"
echo -e "${GREEN}✓ Config saved to $CONFIG_FILE${NC}"
}
# Step 6: User Authorization
authorize() {
echo ""
echo -e "${YELLOW}Step 6: User Authorization${NC}"
echo ""
AUTH_URL="https://login.microsoftonline.com/common/oauth2/v2.0/authorize?client_id=$CLIENT_ID&response_type=code&redirect_uri=$REDIRECT_URI&scope=$(echo $SCOPES | sed 's/ /%20/g')&response_mode=query"
echo "Open this URL in your browser:"
echo ""
echo -e "${BLUE}$AUTH_URL${NC}"
echo ""
echo "After authorizing, you'll be redirected to a page that won't load."
echo "Copy the FULL URL from the address bar and paste it here:"
echo ""
read -p "URL: " REDIRECT_URL
# Extract code from URL
AUTH_CODE=$(echo "$REDIRECT_URL" | grep -oP 'code=\K[^&]+' || echo "")
if [ -z "$AUTH_CODE" ]; then
echo -e "${RED}Could not extract authorization code from URL${NC}"
exit 1
fi
echo ""
echo "Exchanging code for tokens..."
TOKEN_RESPONSE=$(curl -s -X POST "https://login.microsoftonline.com/common/oauth2/v2.0/token" \
-H "Content-Type: application/x-www-form-urlencoded" \
-d "client_id=$CLIENT_ID&client_secret=$CLIENT_SECRET&code=$AUTH_CODE&redirect_uri=$REDIRECT_URI&grant_type=authorization_code&scope=$SCOPES")
if echo "$TOKEN_RESPONSE" | jq -e '.access_token' > /dev/null 2>&1; then
echo "$TOKEN_RESPONSE" > "$CREDS_FILE"
chmod 600 "$CREDS_FILE"
echo -e "${GREEN}✓ Tokens saved to $CREDS_FILE${NC}"
else
echo -e "${RED}Failed to get tokens:${NC}"
echo "$TOKEN_RESPONSE" | jq '.error_description // .'
exit 1
fi
}
# Step 7: Test Connection
test_connection() {
echo ""
echo -e "${YELLOW}Step 7: Testing Connection${NC}"
ACCESS_TOKEN=$(jq -r '.access_token' "$CREDS_FILE")
INBOX=$(curl -s "https://graph.microsoft.com/v1.0/me/mailFolders/inbox" \
-H "Authorization: Bearer $ACCESS_TOKEN")
if echo "$INBOX" | jq -e '.totalItemCount' > /dev/null 2>&1; then
TOTAL=$(echo "$INBOX" | jq '.totalItemCount')
UNREAD=$(echo "$INBOX" | jq '.unreadItemCount')
echo -e "${GREEN}✓ Connection successful!${NC}"
echo ""
echo -e "Inbox: ${BLUE}$TOTAL${NC} emails (${YELLOW}$UNREAD${NC} unread)"
else
echo -e "${RED}Connection test failed${NC}"
echo "$INBOX" | jq '.error.message // .'
exit 1
fi
}
# Main
main() {
check_prereqs
azure_login
create_app
create_secret
add_permissions
save_config
authorize
test_connection
echo ""
echo -e "${GREEN}=== Setup Complete! ===${NC}"
echo ""
echo "You can now use:"
echo " outlook-mail.sh inbox - List emails"
echo " outlook-mail.sh unread - List unread"
echo " outlook-mail.sh search X - Search emails"
echo " outlook-calendar.sh today - Today's events"
echo " outlook-calendar.sh week - This week's events"
echo " outlook-token.sh refresh - Refresh token"
}
main "$@"

65
scripts/outlook-token.sh Normal file
View File

@@ -0,0 +1,65 @@
#!/bin/bash
# Outlook Token Manager
# Usage: outlook-token.sh [refresh|get|test]
CONFIG_DIR="$HOME/.outlook-mcp"
CONFIG_FILE="$CONFIG_DIR/config.json"
CREDS_FILE="$CONFIG_DIR/credentials.json"
# Check if config exists
if [ ! -f "$CONFIG_FILE" ] || [ ! -f "$CREDS_FILE" ]; then
echo "Error: Outlook not configured. Run setup first."
echo "Missing: $CONFIG_FILE or $CREDS_FILE"
exit 1
fi
# Load credentials
CLIENT_ID=$(jq -r '.client_id' "$CONFIG_FILE")
CLIENT_SECRET=$(jq -r '.client_secret' "$CONFIG_FILE")
ACCESS_TOKEN=$(jq -r '.access_token' "$CREDS_FILE")
REFRESH_TOKEN=$(jq -r '.refresh_token' "$CREDS_FILE")
case "$1" in
refresh)
echo "Refreshing token..."
RESPONSE=$(curl -s -X POST "https://login.microsoftonline.com/common/oauth2/v2.0/token" \
-H "Content-Type: application/x-www-form-urlencoded" \
-d "client_id=$CLIENT_ID&client_secret=$CLIENT_SECRET&refresh_token=$REFRESH_TOKEN&grant_type=refresh_token&scope=https://graph.microsoft.com/Mail.ReadWrite https://graph.microsoft.com/Mail.Send https://graph.microsoft.com/Calendars.ReadWrite offline_access")
if echo "$RESPONSE" | jq -e '.access_token' > /dev/null 2>&1; then
echo "$RESPONSE" > "$CREDS_FILE"
echo "Token refreshed successfully"
else
echo "Error refreshing token:"
echo "$RESPONSE" | jq '.error_description // .'
exit 1
fi
;;
get)
echo "$ACCESS_TOKEN"
;;
test)
echo "Testing connection..."
RESULT=$(curl -s "https://graph.microsoft.com/v1.0/me/mailFolders/inbox" \
-H "Authorization: Bearer $ACCESS_TOKEN")
if echo "$RESULT" | jq -e '.totalItemCount' > /dev/null 2>&1; then
TOTAL=$(echo "$RESULT" | jq '.totalItemCount')
UNREAD=$(echo "$RESULT" | jq '.unreadItemCount')
echo "✓ Connected! Inbox: $TOTAL emails ($UNREAD unread)"
else
echo "✗ Connection failed. Try: outlook-token.sh refresh"
echo "$RESULT" | jq '.error.message // .'
exit 1
fi
;;
*)
echo "Usage: outlook-token.sh [refresh|get|test]"
echo " refresh - Refresh the access token"
echo " get - Print current access token"
echo " test - Test the connection"
;;
esac