Skip to main content

Self-hosted iMessage client (Twilio with blue bubbles)

Project description

Lusid

Self-hosted Python interface for iMessage (similar to Twilio) that supports both green and blue bubbles.

Think this is cool? Check out what we're building at lsd.so

Important: Before use, please ensure you are compliant with telecommunications laws in your area and in some cases your recipient's area, including quiet hours, express consent, and other spam prevention laws. Lusid includes support, by default, for users to opt out by texting 'STOP', which is required in many countries including the US.

Requirements

  • Mac running macOS 10.11 or later
  • Python
  • iPhone with SIM (optional, adds SMS/green bubble support)

Installing

Use pip or whichever python package manager you're working with

$ pip install lusid

You'll need to allow your Terminal application (or whichever terminal emulator you're running) to have full disk access in order to view the chat.db file containing your iMessages as well as "Accessibility" permissions via the Security & Privacy settings on your Mac.

Additionally, you'll need to disable System Integrity Protection on your Mac by running this terminal command in recovery mode:

$ csrutil disable

Please note: System Integrity Protection prevents malware from accessing and altering system-level files, and disabling leaves your computer vulnerable to malicious code. We reccomend going through the below article and ensuring you understand the risks before disabling.

For additional information on how to disable System Integrity Protection, the risks associated, and how to re-enable it, refer to this Apple Developer Article.

Quickstart

Sending Messages

If you're just interested in sending a message, import the send_message function and voila

from lusid import send_message

to = "123-456-7890"
body = "Yeehaw"

send_message(to, body)

Reading Messages

If you're just interested in getting iMessage data

from lusid import read_messages

print(read_messages())

If you're interested in not only received but all messages and only ones that were sent in the past 24 hours

from datetime import datetime, timedelta

from lusid import read_messages

print(read_messages(
  since_time=datetime.now() + timedelta(days=-1),
  inbound_only=False,
))

SMS Support

If you're interested in supporting SMS (green bubbles), follow these steps:

  • Ensure your iPhone has an active SIM card and is updated to the latest iOS software.
  • In Settings > Apple ID, sign into the same Apple ID used on your Mac running Lusid.
  • In Settings > Apps > Messages > Text Message Forwarding, enable the Mac running Lusid.

SMS/MMS messages can now be sent by running Lusid normally, provided both your Mac and iPhone are powered on and connected to the internet. To ensure best performance, keep both devices connected to the same network.

Example Usage

Basic

If you're interested in something replying to received messages

  1. Create a "client" to repeatedly read your inbox (the rest of this quickstart assumes you're writing to a file named app.py but feel free to replace that later on with whatever you named your to)
# app.py

from lusid import create_simple_message_client

def start_client():
  create_simple_message_client(
    message_handler=lambda to, body: None,
  )

if __name__ == "__main__":
  start_client()
  1. Define a function for handling messages:
# Snippet

def handle_message(from_number, body):
  print(f"Handling the message [{body}] from [{from_number}]")
  return "Some funny autoreply here" # Or None to not reply at all
  1. Next we're going to include the function we defined earlier
# app.py

from lusid import create_simple_message_client

+def handle_message(from_number, body):
+  print(f"Handling the message [{body}] from [{from_number}]")
+  return "Some funny autoreply here" # Or None to not reply at all

def start_client():
  create_simple_message_client(
    message_handler=lambda to, body: None,
  )

if __name__ == "__main__":
  start_client()

Then actually use it as our message handler

# app.py

from lusid import create_simple_message_client

def handle_message(from_number, body):
  print(f"Handling the message [{body}] from [{from_number}]")
  return "Some funny autoreply here" # Or None to not reply at all

def start_client():
  create_simple_message_client(
-    message_handler=lambda to, body: None,
+    message_handler=handle_message
  )

if __name__ == "__main__":
  start_client()

If you'd like to just copy/paste the resulting code

# app.py

from lusid import create_simple_message_client

def handle_message(from_number, body):
  print(f"Handling the message [{body}] from [{from_number}]")
  return "Some funny autoreply here" # Or None to not reply at all

def start_client():
  create_simple_message_client(
    message_handler=handle_message
  )

if __name__ == "__main__":
  start_client()
  1. Now your script is set up to automatically reply to every received message with "Some funny autoreply here"
$ python app.py
Handling the message [Hello word!] from [+11234567890]

Complex

Suppose you wanted to be able to share cat facts with a specific friend while also having message interaction, here's how you can accomplish that. We'll be adding to the Basic example above

For this particular example we'll be using the python package requests to make a simple API request

$ pip install requests

In short, like how React components have lifecycle methods, the message client features a handle_post_read method that can be specified at instantiation.

# app.py

+import random
+from requests import get
from lusid import create_simple_message_client

def handle_message(from_number, body):
  print(f"Handling the message [{body}] from [{from_number}]")
  return "Some funny autoreply here" # Or None to not reply at all

+def handle_post_read(cls):
+  facts = get("https://cat-fact.herokuapp.com/facts").json()
+  fact = random.choice(facts)["text"]
+
+  print(f"Telling kevin that {fact}")
+
+  kevin = "123-456-7890"
+  cls.send_message(kevin, fact)

def start_client():
  create_simple_message_client(
    message_handler=handle_message,
+    handle_post_read=handle_post_read
  )

if __name__ == "__main__":
  start_client()

Or, if you'd like to just copy and paste

# app.py

import random
from requests import get
from lusid import create_simple_message_client

def handle_message(from_number, body):
  print(f"Handling the message [{body}] from [{from_number}]")
  return "Some funny autoreply here" # Or None to not reply at all

def handle_post_read(cls):
  facts = get("https://cat-fact.herokuapp.com/facts").json()
  fact = random.choice(facts)["text"]

  print(f"Telling kevin that {fact}")

  kevin = "123-456-7890"
  cls.send_message(kevin, fact)

def start_client():
  create_simple_message_client(
    message_handler=handle_message,
    handle_post_read=handle_post_read
  )

if __name__ == "__main__":
  start_client()

And here's what it looks like when run

$ python app.py
Telling kevin that Cats are the most popular pet in the United States: There are 88 million pet cats and 74 million dogs.
Telling kevin that Most cats are lactose intolerant, and milk can cause painful stomach cramps and diarrhea. It's best to forego the milk and just give your cat the standard: clean, cool drinking water.
Telling kevin that Owning a cat can reduce the risk of stroke and heart attack by a third.

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

lusid-1.0.10.tar.gz (16.8 kB view details)

Uploaded Source

Built Distribution

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

lusid-1.0.10-py3-none-any.whl (19.0 kB view details)

Uploaded Python 3

File details

Details for the file lusid-1.0.10.tar.gz.

File metadata

  • Download URL: lusid-1.0.10.tar.gz
  • Upload date:
  • Size: 16.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.1.1 CPython/3.12.3

File hashes

Hashes for lusid-1.0.10.tar.gz
Algorithm Hash digest
SHA256 79707f9b573a8fe626ffbec80ce1ad81ce77d1a4b59ff93ea37fda57e1188844
MD5 140b31c5a8f49a7378cfb342e17281f9
BLAKE2b-256 f63d4649c7d17908c6c41e2ad58a7e44d08ec0a90c2ddc9732ee9567712ef01b

See more details on using hashes here.

File details

Details for the file lusid-1.0.10-py3-none-any.whl.

File metadata

  • Download URL: lusid-1.0.10-py3-none-any.whl
  • Upload date:
  • Size: 19.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.1.1 CPython/3.12.3

File hashes

Hashes for lusid-1.0.10-py3-none-any.whl
Algorithm Hash digest
SHA256 8b8dab18e75497bb21c856b9ebb42bb79627d9e54cabeffe561ff3f9854981f0
MD5 f6a7ab63b55a247336619d58b49146f3
BLAKE2b-256 4a89f8f4fc4b22ec4c2895920119a997996d981033833028806b953e30ab1aee

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