Docker container to expose a local RTMP, RTSP, and HLS stream for ALL your Wyze cameras including the outdoor and doorbell cams. No third-party or special firmware required.
It just works!
Based on @noelhibbard's script with kroo/wyzecam and aler9/rtsp-simple-server.
Please consider supporting this project if you found it useful.
Install docker and use your Wyze credentials to run:
docker run \
-e WYZE_EMAIL=you@email.com \
-e WYZE_PASSWORD=yourpassw0rd \
-p 8888:8888 mrlt8/wyze-bridge:latest
You can view your stream by visiting: http://localhost:8888/cam-nickname
where localhost is the hostname or ip of the machine running the bridge followed by the cam nickname in lowercase with -
in place of spaces.
See basic usage for additional information.
Audio is also coming soon. Please check out the audio branch to report any issues.
- Support for Wyze Cam Outdoor v2! (#354) Thanks @urubos!
- Fixed bug where the add-on would not start in Home Assistant if hostname was not set. (#355) Thanks @cbrightly!
- Fixed bug where rtsp-simple-server would refuse connections if the camera name contained a special character. (#356) Thanks @JochenKlenk!
- Set defualt doorbell bitrate to 180.
There is a known bug/issue with certain doorbells that drift out of sync due to the day/night fps change (#340).
Camera | Model | Supported |
---|---|---|
Wyze Cam v1 | WYZEC1 | |
Wyze Cam V2 | WYZEC1-JZ | ✅ |
Wyze Cam V3 | WYZE_CAKP2JFUS | ✅ |
Wyze Cam Floodlight | WYZE_CAKP2JFUS | ✅ |
Wyze Cam Pan | WYZECP1_JEF | ✅ |
Wyze Cam Pan v2 | HL_PAN2 | ✅ |
Wyze Cam Outdoor | WVOD1 | ✅ |
Wyze Cam Outdoor v2 | HL_WCO2 | ✅ |
Wyze Cam Doorbell | WYZEDB3 | ✅ |
Wyze Cam Doorbell Pro | GW_BE1 | ❓ |
The bridge should be compatible with official firmware from wyze.
Should work on most x64 systems as well as on some arm-based systems like the Raspberry Pi.
The container can be run on its own, in Portainer, Unraid, or as a Home Assistant Add-on.
This is similar to the docker run command, but will save all your options in a yaml file.
- Install Docker Compose.
- Use the sample as a guide to create a
docker-compose.yml
file with your wyze credentials. - Run
docker-compose up
.
Once you're happy with your config you can use docker-compose up -d
to run it in detached mode.
To update your container, cd
into the directory where your docker-compose.yml
is located and run:
docker-compose pull # Pull new image
docker-compose up -d # Restart container in detached mode
docker image prune # Remove old images
Visit the wiki page for additional information on Home Assistant.
- Two-Step Verification
- ARM/Raspberry Pi
- LAN mode
- Portainer
- Unraid
- Home Assistant
- Homebridge Camera FFmpeg
- HomeKit Secure Video
Audio is coming soon.
If your email or password contains a %
or $
character, you may need to escape them with an extra character. e.g., pa$$word
should be entered as pa$$$$word
By default, the bridge will create three streams for each of your cameras which can be accessed at the following URIs, where camera-nickname
is the name of the camera set in the Wyze app and converted to lower case with hyphens in place of spaces. e.g. 'Front Door' would be /front-door
Replace localhost with the hostname or ip of the machine running the bridge:
-
RTMP:
rtmp://localhost:1935/camera-nickname
-
RTSP:
rtsp://localhost:8554/camera-nickname
-
HLS:
http://localhost:8888/camera-nickname/stream.m3u8
-
HLS can also be viewed in the browser using:
http://localhost:8888/camera-nickname
Two-factor authentication ("Two-Step Verification" in the wyze app) is supported and will automatically be detected, however additional steps are required to enter your verification code.
-
Echo the verification code directly to
/tokens/mfa_token
by opening a second terminal window and using:docker exec -it wyze-bridge sh -c 'echo "123456" > /tokens/mfa_token'
-
Mount
/tokens/
locally and add your verification code to a file namedmfa_token
:volumes: - ./tokens:/tokens/
You can also have the bridge auto generate and enter a Time-based One-Time Password (TOTP) by adding the secret key to the file /tokens/totp
on standard installs or /config/wyze-bridge/totp
for Home Assistant installs. You will need to create the file if it doesn't exist and mount it if necessary.
-
🏠 Home Assistant:
Add your code to the text file:
/config/wyze-bridge/mfa_token.txt
. -
Portainer:
Use the console to echo your code to the container:
echo "123456" > /tokens/mfa_token
WYZE_EMAIL and WYZE_PASSWORD are the only two required environment variables. The following envs are all optional.
The default option will automatically create a stream for all the cameras on your account, but you can use the following environment options in your docker-compose.yml
to filter the cameras.
All options are case-insensitivE, and take single or comma separated values.
-
Whitelist by Camera Name (set in the wyze app):
environment: .. - FILTER_NAMES=Front Door, Driveway, porch cam
-
Whitelist by Camera MAC Address:
- FILTER_MACS=00:aA:22:33:44:55, Aa22334455bB
-
Whitelist by Camera Model:
- FILTER_MODELS=WYZEC1-JZ
-
Whitelist by Camera Model Name:
- FILTER_MODELS=V2, v3, Pan
-
Blacklisting:
You can reverse any of these whitelists into blacklists by setting
FILTER_BLOCK
.environment: .. - FILTER_NAMES=Bedroom - FILTER_BLOCK=true
Like the wyze app, the tutk library has three different modes to connect to the camera and will attempt to stream directly from the camera when on the same LAN as the camera in "LAN mode". If the camera is not available locally, it will either attempt to stream directly from your network using "P2P Mode" or relay the stream via the wyze servers (AWS) in "relay mode".
LAN mode is more ideal as all streaming will be local and won't use additional bandwidth.
By default, the bridge will attempt to connect via "LAN Mode", but will fallback to other methods if LAN mode fails.
You can restrict streaming to LAN only by setting the NET_MODE=LAN
environment variable:
environment:
..
- NET_MODE=LAN
NET_MODE=P2P
is ideal when running the bridge remotely on a different network or on a VPS and will allow the bridge to stream directly from the camera over the internet while blocking "Relay Mode".
NET_MODE=ANY
is the equivalent to leaving NET_MODE
unset and will allow the connection to fallback to P2P or relay mode.
In the event that you need to allow the bridge to access a select number of cameras outside of your LAN, you can specify them by appending the camera name to NET_MODE
, where CAM_NAME
is the camera name in UPPERCASE and _
in place of spaces and hyphens:
..
- NET_MODE=LAN
- NET_MODE_CAM_NAME=P2P
-
SNAPSHOT=API
Will run ONCE at startup and will grab a high-quality thumbnail from the wyze api and save it to/img/cam-name.jpg
on docker installs or/config/www/cam-name.jpg
in Home Assistant mode. -
SNAPSHOT=RTSP
Will run every 180 seconds (configurable) and wll grab a new frame from the RTSP stream every iteration and save it to/img/cam-name.jpg
on standard docker installs or/config/www/cam-name.jpg
in Home Assistant mode. Can specify a custom interval withSNAPSHOT=RTSP(INT)
e.g.SNAPSHOT=RTSP30
to run every 30 seconds -
IMG_DIR=/img/
Specify the directory where the snapshots will be saved within the container. Use volumes in docker to map to an external directory. -
IMG_TYPE
Specify the file type of the image, e.g.IMG_TYPE=png
. Will default to jpg.
The bridge can be configured to record all or select camera streams to the container which can be mapped to a local directory.
environment:
...
- TZ=America/New_York
- RECORD_ALL=True
volumes:
- /local/path/:/record/
Or to specify select cameras, where CAM_NAME
is the camera name in UPPERCASE and _
in place of spaces and hyphens:
- RECORD_CAM_NAME=True
- RECORD_OTHER_CAM=True
See the Stream Recording wiki page page for additional options.
Basic livestream support is available for YouTube and Facebook, but you can also specify any custom rtmp server for other services like Twitch.
To use this feature, set a new env in your docker-compose.yml with the service (YOUTUBE_
or FACEBOOK_
) prefix followed by the camera name in UPPERCASE with _
in place of spaces and hyphens, and set your stream key as the value. Custom rtmp servers can be specified using the LIVESTREAM_
prefix:
- YOUTUBE_FRONT_DOOR=MY-STREAM-KEY
- FACEBOOK_OTHER_CAM=MY-STREAM-KEY
# twitch example:
- LIVESTREAM_CAM_NAME=rtmp://jfk.contribute.live-video.net/app/MY-STREAM-KEY
Some basic MQTT support is now available in v0.7.0.
MQTT auth and discovery should be automatic in Home Assistant mode - can be disabled by setting MQTT_HOST
to False.
ENV Name | Description | Example |
---|---|---|
MQTT_HOST | IP/Hostname AND Port of the MQTT broker | core-mosquitto:1883 |
MQTT_AUTH | Username AND password; leave blank if none | user:pass |
MQTT_TOPIC | Optional - Specify topic prefix | myhome |
MQTT_DTOPIC | Optional - Discovery topic for home assistant | homeassistant |
Bitrate and resolution of the stream from the wyze camera can be adjusted with:
environment:
- QUALITY=HD120
where CAM_NAME
is the camera name in UPPERCASE and _
in place of spaces and hyphens:
environment:
- QUALITY_CAM_NAME=HD120
Additional info:
- Resolution can be set to
SD
(360p in the app) orHD
- 640x360/1920x1080 for cams or 480x640/1296x1728 for doorbells. - Bitrate can be set from 30 to 255. Some bitrates may not work with certain resolutions.
- Adjusting the bitrate and resolution in the bridge will also change the stream in the wyze app and vice versa.
- App equivalents would be:
- 360p - SD30
- SD - HD60 (HD120 for doorbell)
- HD - HD120 (HD180 for doorbell)
You can pass a custom command to FFmpeg by using FFMPEG_CMD
in your docker-compose.yml:
environment:
..
- FFMPEG_CMD=-f h264 -i - -vcodec copy -f flv rtmp://rtsp-server:1935/{cam_name}
where CAM_NAME
is the camera name in UPPERCASE and _
in place of spaces and hyphens:
- FFMPEG_CMD_CAM_NAME=ffmpeg -f h264 -i - -vcodec copy -f flv rtmp://rtsp-server:1935/{cam_name}
Additional info:
- The
ffmpeg
command is implied and is optional. - The camera name is available as a variable
{cam_name}
for lowercase and{CAM_NAME}
for uppercase.
Custom ffmpeg flags can easily be tested with:
environment:
..
- FFMPEG_FLAGS=-fflags +flush_packets+genpts+discardcorrupt+nobuffer
or where CAM_NAME
is the camera name in UPPERCASE and _
in place of spaces and hyphens:
- FFMPEG_FLAGS_CAM_NAME=-flags low_delay
rtsp-simple-server options can be customized as an environment variable in your docker-compose.yml by prefixing RTSP_
to the UPPERCASE parameter.
e.g. use - RTSP_RTSPADDRESS=:8555
to overwrite the default rtspAddress
.
or - RTSP_PATHS_ALL_READUSER=123
to customize a path specific option like paths: all: readuser:123
For camera specific options with spaces in the name of the camera (e.g. Front Door
), be sure to replace the spaces with the URI_SEPARATOR
which defaults to -
. So Front Door
would be represented as FRONT-DOOR
, and paths: Front Door: runOnReady: ffmpeg...
could be set in your docker-compose as:
environment:
...
- RTSP_PATHS_FRONT-DOOR_RUNONREADY=ffmpeg...
environment options:
-
FRESH_DATA
(bool) Remove local cache and pull new data from wyze servers. -
URI_SEPARATOR
(-|_|#) Customize the separator used to replace spaces in the URI; available values are-
,_
, or use#
to remove spaces. -
CONNECT_TIMEOUT
(int) Adjust the number of seconds to wait before timing out when connecting to camera. Default:15
-
KEEP_BAD_FRAMES
(bool) Keep frames that may be missing a keyframe or preceding frames. -
IGNORE_OFFLINE
(bool) Ignore offline cameras until container restarts. -
OFFLINE_TIME
(int) Customize the sleep time when a camera is offline. Default:10
-
DEBUG_FRAMES
(bool) Show all lost/incomplete frames. -
DEBUG_LEVEL
(debug|info|warning|error) Adjust the level of upstream logging. -
RTSP_READTIMEOUT
(str) Adjust the max number of seconds of missing frames allowed before a stream is restarted. Be sure to include the s after the number. Default:20s
-
RTSP_LOGLEVEL
(debug|info|warn) Adjust the verbosity of rtsp-simple-server; available values are "warn", "info", "debug". -
SKIP_RTSP_LOG
(bool) Prevent "read" spam in the logs when using RTSP based snapshots. Works by only logginng clients that stay connected for longer than 3s. -
DEBUG_FFMPEG
(bool) Enable additional logging from FFmpeg. -
FORCE_FPS_CAM_NAME
(int) Force a specific camera to use a different FPS, whereCAM_NAME
is the camera name in UPPERCASE and_
in place of spaces and hyphens. -
FPS_FIX
(bool) Set camera parameter to match the actual FPS being sent by the camera. Potential fix slow/fast SD card and cloud recordings. -
WEBRTC
(bool) Display WebRTC credentials for cameras.