Skip to main content

A fastai-style Gmail API client with convenient read/write support

Project description

solvemail

A simple Gmail / Google Workspace email client built on the official Gmail API, using the fastai/fastcore coding style.

Install

pip install solvemail

Or for development:

pip install -e .

OAuth setup

For detailed instructions on setting up Google Cloud credentials, see ezgmail's excellent documentation.

In brief:

  1. Create an OAuth Client ID (Desktop app) in Google Cloud Console and enable the Gmail API
  2. Download the client secrets JSON as credentials.json
  3. Put credentials.json next to your script (or pass its path)

On first run, solvemail will open a browser to authorize and will save token.json.

Quick start

import solvemail

solvemail.init()  # reads credentials.json + token.json in cwd

# For multiple accounts, use separate token files:
# solvemail.init(token_path='work.json')    # first run opens browser to auth
# solvemail.init(token_path='personal.json') # switch account without re-auth

# Check which account you're using
solvemail.profile().email

# solvemail exports the key API functionality through wildcard import
from solvemail import *

# Search for threads
threads = search_threads('is:unread newer_than:7d', max_results=10)

# Get thread details
t = threads[0]
for m in t.msgs():
    print(m.frm, '|', m.subj)

# Read a message
m = t.msgs()[0]
m.subj, m.frm, m.snip, m.text()

# Send an email
send(to='someone@example.com', subj='Hello', body='Hi there!')

# Create and send a reply
draft = t.reply_draft(body='Thanks!')
draft.send()

Features

Searching

# Search threads (conversations)
search_threads('from:boss@company.com', max_results=20)

# Search individual messages
search_msgs('has:attachment filename:pdf', max_results=100)

Messages

m = msg(id)           # Fetch by id
m.subj, m.frm, m.to             # Headers
m.text(), m.html()              # Body content
m.mark_read(), m.mark_unread()  # Read status
m.star(), m.unstar()            # Starred
m.archive()                     # Remove from inbox
m.trash(), m.untrash()          # Trash
m.add_labels('MyLabel')         # Add labels
m.rm_labels('INBOX')            # Remove labels

Threads

t = thread(id)        # Fetch by id
t.msgs()                        # List messages
t[0], t[-1]                     # Index into messages
t.reply_draft(body='...')       # Create reply draft
t.reply(body='...')             # Send reply directly

# Batch fetch multiple threads efficiently (one HTTP call)
threads = search_threads('in:inbox', max_results=50)
threads = get_threads(threads)

Message display

Messages render nicely in Jupyter notebooks (quotes and signatures stripped automatically).

m = t[-1]
m.body()   # Cleaned text (no quotes/signatures)
m.html()   # HTML body (falls back to text wrapped in <pre>)

# View message with headers (as dict or plain text)
view_msg(m.id)                      # Returns dict with headers + body
view_msg(m.id, as_json=False)       # Returns formatted text

# View full thread
view_thread(t.id)                   # Dict of msgid -> msg dict
view_thread(t.id, as_json=False)    # Concatenated text with separators

Inbox helpers

view_inbox(max_msgs=20)             # Batch fetch inbox messages
view_inbox_threads(max_threads=20)  # Batch fetch inbox threads
view_inbox(unread=True)             # Only unread

Labels

labels()                        # List all labels
label('INBOX')                  # Get by name or id
find_labels('project')          # Search labels
create_label('My Label')        # Create new label

Drafts

drafts()                        # List drafts
create_draft(to='...', subj='...', body='...')
reply_to_thread(thread_id, body='...')

Bulk operations

# Batch modify labels (auto-chunks, no 1000 message limit)
ids = [m.id for m in search_msgs('in:inbox')]
batch_label(ids, add=['SPAM'], rm=['INBOX'])

# Trash multiple messages
trash_msgs(ids)

# Permanently delete (requires full mail scope)
batch_delete(ids)

Using the Gmail class directly

from solvemail import Gmail

g = Gmail(creds_path='credentials.json', token_path='token.json')
lbl = g.create_label('test-label')

m = g.send(to=g.profile().email, subj='hello', body='hi there')
m.add_labels(lbl)

lbl.delete()

Testing

Set these env vars to run e2e tests against a throwaway Gmail/Workspace account:

  • GMAILX_CREDS — path to credentials.json
  • GMAILX_TOKEN — path to token.json (will be created if missing)
  • GMAILX_E2E — set to 1 to enable e2e tests
pytest -q

Credits

Inspired by ezgmail by Al Sweigart — thanks Al for the great work! The ezgmail repo also has excellent documentation on setting up Gmail API credentials.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

solvemail-0.1.3.tar.gz (14.5 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

solvemail-0.1.3-py3-none-any.whl (14.3 kB view details)

Uploaded Python 3

File details

Details for the file solvemail-0.1.3.tar.gz.

File metadata

  • Download URL: solvemail-0.1.3.tar.gz
  • Upload date:
  • Size: 14.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.8

File hashes

Hashes for solvemail-0.1.3.tar.gz
Algorithm Hash digest
SHA256 0f0f3fb4bf55c56caa6f27b863f9ac3f5d9376ec6083c7b94da09211195a5a10
MD5 d0ff1e3f13e719f4d940b4dc7cd60aa3
BLAKE2b-256 77f8ceb41cf96f765ce887f907b06b7781f2578d76cdab61ed38d220977cf736

See more details on using hashes here.

File details

Details for the file solvemail-0.1.3-py3-none-any.whl.

File metadata

  • Download URL: solvemail-0.1.3-py3-none-any.whl
  • Upload date:
  • Size: 14.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.8

File hashes

Hashes for solvemail-0.1.3-py3-none-any.whl
Algorithm Hash digest
SHA256 1a672a58badf69556bad07cd8a9269b85513f2649bef4a56a2d6956c764056b2
MD5 a5c88fdaa1bb79faec21c5dc9b96e8f9
BLAKE2b-256 d6eedeb28d70e7b77564f7687de2045af4dd9e404834ec8e5eafe06d0c02d404

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page