Lack of data validation - Path Traversal In motioneye

Description

motionEye: LFI → pass‑the‑hash admin → unsafe restore → unauth action exec (RCE)

Summary

A multi‑stage chain in motionEye leads to remote code execution. The chain combines:

    Arbitrary file read (LFI) via the picture download endpoint for local motion cameras using absolute paths.

    Pass‑the‑hash admin auth due to accepting request signatures computed with password hashes.

    Unsafe config restore that extracts attacker‑controlled tarballs into CONF_PATH.

    Unauthenticated action execution via /action/<id>/<action>.

If the normal user password is unset, the chain becomes unauthenticated RCE. If a normal password exists, a normal user can still achieve admin escalation and RCE.


Affected Code (motionEye repo)

1) LFI (absolute path) — picture/<id>/download

Files:

    motioneye/motioneye/handlers/picture.pydownload() (local motion camera branch)

    motioneye/motioneye/mediafiles.pyget_media_content()

Issue: get_media_content() only blocks .. and then joins target_dir with path. Absolute paths (e.g. /etc/hosts) bypass the join and are read directly.

2) Pass‑the‑hash admin auth

File: motioneye/motioneye/handlers/base.pyget_current_user()

Issue: The signature check allows signatures computed using the admin password hash (SHA1) as the key. If the hash is leaked (via LFI), admin access can be obtained without the plaintext password.

3) Unsafe restore (tar extraction)

File: motioneye/motioneye/config.pyrestore()

Issue: tar zxC CONF_PATH is used on user‑supplied data without sanitizing entries. A crafted tar can drop executable files into CONF_PATH.

4) Unauthenticated action execution

File: motioneye/motioneye/handlers/action.pypost()

Issue: No authentication decorator is present. It executes <action>_<camera_id> found in CONF_PATH with subprocess.Popen.


Exploit Chain (Detailed)

    Create or find a local motion camera id (local motion cameras are required for the vulnerable LFI path).

    LFI via picture download:

      Request: /picture/<id>/download/<absolute_path>

      Example: /picture/1/download/%2Fetc%2Fhosts

      Result: Arbitrary file read.

    Read admin hash from /etc/motioneye/motion.conf:

      Contains @admin_password <SHA1_HASH>.

    Pass‑the‑hash admin:

      Compute signature for /config/restore?_username=admin using the hash as key.

      Admin access is accepted with hash‑based signatures.

    Restore malicious tar:

      Upload a tar containing lock_<id> (or any action) as an executable.

      File is written into CONF_PATH by restore.

    Trigger unauth action:

      POST /action/<id>/lock

      The server executes the injected file.


Proof of Execution (Observed Output)

In local testing, the injected action created a marker file:

/tmp/meye_rce_ok

Verification command:

docker exec -it motioneye ls -la /tmp | grep meye_rce_ok

Example output:

-rw-r--r-- 1 root root 0 ... /tmp/meye_rce_ok

Preconditions / Requirements

    At least one local motion camera exists (e.g., netcam_url, videodevice).

    picture/<id>/download is reachable:

      Unauth if @normal_password is empty (default in some installs).

      Auth required if normal password is set (attacker needs normal creds).


Impact

    Unauth RCE (normal password unset).

    Authenticated RCE (normal user → admin → RCE).

    Arbitrary file read on server filesystem.

    Full compromise of motionEye process account.


Suggested Fixes

    Block absolute paths in get_media_content() and get_media_path().

    Remove hash‑based signature acceptance; only accept signatures computed with plaintext passwords.

    Harden restore: reject absolute paths, .., symlinks, non‑regular files.

    Require authentication on ActionHandler (admin‑only).

Mitigation

Update Impact

Minimal update. May introduce new vulnerabilities or breaking changes.

Ecosystem
Package
Affected version
Patched versions
FLAT-DK30J – Vulnerability | Fluid Attacks Database