Skip to main content

Bridge Android camera media into the macOS / iCloud Photos library via an always-on Mac.

Project description

icsync

PyPI Python License: MIT Release

Sync Android camera photos & videos into your iCloud Photo Library — as full-resolution originals, in the real Photos app, not some third-party gallery.

There is no public Apple API to write into the iCloud Photo Library. icsync uses the only robust path: an always-on Mac signed into iCloud, importing media via osxphotos and letting Photos sync it up.

Android DCIM/Camera ──Syncthing send-only──▶ Mac inbox
        Mac inbox ──icsync (launchd)──▶ osxphotos import ──▶ Photos.app ──▶ iCloud Photos

icsync is the Mac-side bridge: a small CLI that watches an inbox folder, imports stable media into Photos, archives the originals, and installs itself as a launchd agent. The Android→Mac transport is off-the-shelf (Syncthing — no port-forwarding, no exposing the Mac).

Install

Needs pipx (brew install pipx). Install the CLI — it pulls in osxphotos automatically:

pipx install icsync

No pipx? python3 -m pip install --user icsync works too.

Then set up the bridge in order:

icsync doctor     # 1. check osxphotos + reminders (do prerequisites below first)
icsync install    # 2. create dirs + load the LaunchAgent (runs every 60s)
icsync status     # 3. confirm it's loaded and watching the inbox

Run icsync install after the prerequisites below — the agent imports into Photos and needs iCloud Photos on plus the one-time macOS permission grants.

From source:

git clone https://github.com/akurach/icsync && cd icsync
pipx install .

Prerequisites

  1. Photos.app signed into your Apple ID, iCloud Photos ON (Photos → Settings → iCloud).
  2. Syncthing on the Mac with a Receive Only folder at the icsync inbox (~/Documents/icsync/data/inbox by default), paired to the Android device's Send Only folder on DCIM/Camera.
  3. On first import macOS will ask to grant Automation → Photos; osxphotos may also need Full Disk Access (System Settings → Privacy & Security).

Commands

Command Does
icsync install Create data dirs, write + load the LaunchAgent.
icsync uninstall Unload + remove the LaunchAgent.
icsync start / stop Load / unload the agent.
icsync status Show install state, pending media, paths.
icsync run Run one import cycle (what launchd calls).
icsync watch Loop cycles in the foreground (no launchd).
icsync doctor Check osxphotos, dirs, agent, reminders.

How it behaves

  • Idempotentosxphotos import --skip-dups; re-delivered files don't duplicate.
  • Safe — imported originals are moved to data/archive/<date>/, not deleted; a retention prune removes them after ICSYNC_RETENTION_DAYS (default 7). Set it high to keep originals indefinitely.
  • One-way — Syncthing send-only → receive-only; nothing flows back to the phone.
  • Mac off = delay, not loss — Syncthing on Android queues and retries; icsync catches up when the Mac returns.
  • Partial-write safe — only files older than ICSYNC_STABLE_SECONDS (default 20) are imported, so half-synced files are skipped until complete.

Configuration (env vars)

Var Default Meaning
ICSYNC_ROOT ~/Documents/icsync/data Base data dir.
ICSYNC_INBOX …/inbox Watched folder (Syncthing target).
ICSYNC_ARCHIVE …/archive Where originals land after import.
ICSYNC_STABLE_SECONDS 20 Min file age before import.
ICSYNC_RETENTION_DAYS 7 Archive purge age.
ICSYNC_OSXPHOTOS autodetect Explicit path to the osxphotos binary.

To override for the agent, add them to the EnvironmentVariables dict in ~/Library/LaunchAgents/com.icsync.import.plist and icsync start.

Limitations

  • Requires an always-on Mac as the bridge — no Apple device means no reliable path.
  • Android "motion photos" import as stills (no Apple Live Photo pairing).
  • Not affiliated with Apple; uses the public Photos import surface only.

License

MIT

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

icsync-0.1.1.tar.gz (9.8 kB view details)

Uploaded Source

Built Distribution

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

icsync-0.1.1-py3-none-any.whl (11.2 kB view details)

Uploaded Python 3

File details

Details for the file icsync-0.1.1.tar.gz.

File metadata

  • Download URL: icsync-0.1.1.tar.gz
  • Upload date:
  • Size: 9.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for icsync-0.1.1.tar.gz
Algorithm Hash digest
SHA256 a91aff36b5e81eb67852e6d2876470ec56bd7ae2aaa30a44dbd1e49b6e270398
MD5 372dd1387a629402ace190fe8a38db60
BLAKE2b-256 e509b8a27ea42aa7a2983e7cb0c00fad89d7374b69a50703ecd95c669c2ec4fb

See more details on using hashes here.

Provenance

The following attestation bundles were made for icsync-0.1.1.tar.gz:

Publisher: release.yml on akurach/icsync

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file icsync-0.1.1-py3-none-any.whl.

File metadata

  • Download URL: icsync-0.1.1-py3-none-any.whl
  • Upload date:
  • Size: 11.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for icsync-0.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 1d86611c50c4837fda4a55c8eb675e36ced243fc806e2f1ce26158814b0fe24b
MD5 ffddb1b68eb5bbc9062657bebf508fe0
BLAKE2b-256 19c0d0226ac9b03ab02582211a8cd31bf10f38fe6a56d8cc018a54f16d7a4d39

See more details on using hashes here.

Provenance

The following attestation bundles were made for icsync-0.1.1-py3-none-any.whl:

Publisher: release.yml on akurach/icsync

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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