/** * Completely Unofficial Ring Connect App For Floodlights/Spotlights/Chimes Only (Don't hate me, Ring guys. I had to do it.) * * Copyright 2019-2020 Ben Rimmasch * Copyright 2022 Caleb Morse * * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except * in compliance with the License. You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License * for the specific language governing permissions and limitations under the License. */ // @todo Other urls that could maybe be used for something // APP_API_BASE_URL + '/mode/location/${location_id}/settings' // APP_API_BASE_URL + '/mode/location/${location_id}/sharing' // CLIENT_API_BASE_URL + '/profile' import com.hubitat.app.ChildDeviceWrapper import groovy.json.JsonOutput import groovy.transform.Field import javax.net.ssl.SSLHandshakeException import javax.net.ssl.SSLPeerUnverifiedException import groovyx.net.http.ResponseParseException import org.apache.http.conn.ConnectTimeoutException import org.apache.http.conn.HttpHostConnectException definition( name: "Unofficial Ring Connect", namespace: "ring-hubitat-codahq", author: "Ben Rimmasch (codahq)", description: "Service Manager for Ring Alarm, Smart Lighting, Floodlights, Spotlights, Chimes, Cameras, and Doorbells", category: "Convenience", singleInstance: true, // Hubitat documentation says these aren't used and should just be set to an empty string iconUrl: "", iconX2Url: "", iconX3Url: "", ) preferences { page(name: "mainPage") page(name: "login") page(name: "secondStep") page(name: "authCheck") page(name: "locations") page(name: "addDevices") page(name: "deviceDiscovery") page(name: "notifications") page(name: "ifttt") page(name: "pollingPage") page(name: "snapshots") } def login() { app.removeSetting('twoStepCode') dynamicPage(name: "login", title: "Log into Your Ring Account", nextPage: "secondStep", uninstall: true) { section("Ring Account Information") { paragraph "

NOTE: Ring now requires two-factor authentication. You will be prompted for your code on the next page." input "username", "email", title: "Ring Username", description: "Email used to login to Ring.com", required: true input "password", "password", title: "Ring Password", description: "Password you login to Ring.com", required: true } } } def secondStep() { unschedule() if (!authPassword() && state.authResponse != "challenge") { return dynamicPage(name: "secondStep", title: "Authenticate failed!", nextPage: "login", uninstall: true) { section { paragraph "Authentication error: ${state.authResponse}" } } } dynamicPage(name: "secondStep", title: "Check text messages or email for the 2-step authentication code", nextPage: "authCheck", uninstall: true) { section("2-Step Code") { input "twoStepCode", "password", title: "Code", description: "2-Step Temporary Code", required: true } } } def authCheck() { // If auth succeeded, do a quick api request to verify we're authenticated if (authTwoFactorChallenge(twoStepCode) && apiRequestLocations()) { dynamicPage(name: "authCheck", title: "Authentication successful", nextPage: "locations", uninstall: true) { section { paragraph "Authentication successful. Click next to continue" } } } else { dynamicPage(name: "authCheck", title: "Authentication failed", nextPage: "login", uninstall: true) { section { paragraph "Authentication failed. Error message was: ${state.authResponse}" paragraph "Click next to try again" } } } } def locations() { state.locationOptions = [:] apiRequestLocations()?.each { state.locationOptions[it.location_id.toString()] = it.name.toString() } if (!state.locationOptions) { return dynamicPage(name: "locations", title: "Not authenticated", nextPage: "login", uninstall: true) { section("Please check your Ring username and password") { paragraph "" } } } dynamicPage(name: "locations", title: "Select which location you want to use", nextPage: "mainPage", uninstall: true) { section("Locations") { input "selectedLocations", "enum", required: true, title: "Select a location", options: state.locationOptions } } } def mainPage() { final boolean isLoggedIn = loggedIn() if (tokenReset) { app.updateSetting("tokenReset", false) state.accessToken = null } dynamicPage(name: "mainPage", title: "Manage Your Ring Devices", uninstall: true, install: true) { if (!state.accessToken) { // Access token for Ecobee to make a callback into this code try { state.accessToken = createAccessToken() } catch (final Exception e) { log.error("mainPage - OAuth Exception: ${e}") section('

OAuth Error

') { paragraph('OAUTH is not currently enabled for this app. You must enable it to continue. You can do this from the "Apps Code" section of the Hubitat UI for this app (the Hubitat "Official Ring Connect"). Enabling oauth should not be necessary if installed with HPM') } } } section("Ring Account Information (${isLoggedIn ? 'Successfully Logged In!' : 'Not Logged In. Please Configure!'})") { href "login", title: "Log into Your Ring Account" } if (isLoggedIn) { final Map location = getSelectedLocation() logTrace "location: $location" if (location) { if (!getAPIDevice(location)) { section("There was an issue finding/migrating your API device! Please check the logs!") { paragraph "" } } section("Configure Devices For Location: ${location.name}") { href "deviceDiscovery", title: "Discover Devices" } } else { section("Error getting selected locations") { paragraph "" } } } section("Getting Events From Ring") { href "notifications", title: "Configure the way that Hubitat will get motion alert and ring events" } section("Camera Thumbnail Images") { href "snapshots", title: "Configure the way that Hubitat will get camera thumbnail images" } List childDevs = getChildDevices() if (childDevs) { section("Managed Installed Child Devices", hidden: true, hideable: true, hideWhenEmpty: true) { for (ChildDeviceWrapper child in childDevs.sort { a, b -> a.deviceNetworkId <=> b.deviceNetworkId }) { String description = "Click here to manage this device" if (child.deviceNetworkId.startsWith(RING_API_DNI)) { description += "\nThis device manages the WebSockets connection for your Ring Alarm Hub/Lighting Bridge. To create children of the Ring Alarm Hub/Lighting Bridge, you must use this device's \"createDevices\" command device" } href child.deviceNetworkId, description: description, title: child.label, url: "/device/edit/${child.id}" } } } section("Configure Logging", hidden: true, hideable: true) { input name: "descriptionTextEnable", type: "bool", title: "Enable descriptionText logging", defaultValue: false input name: "logEnable", type: "bool", title: "Enable debug logging", defaultValue: false input name: "traceLogEnable", type: "bool", title: "Enable trace logging", defaultValue: false } section("Advanced options", hidden: true, hideable: true) { paragraph('Reset Compromised OAuth Access Token') paragraph('Do not toggle this button without understanding the following. Resetting this token will require you to manually update ALL of the URLs in any existing dashboard tile any IFTTT applet. There is no need to reset the token unless it was compromised.') input name: "tokenReset", type: "bool", title: "Toggle this to reset your app's OAuth token", defaultValue: false, submitOnChange: true } donationPageSection() } } def notifications() { dynamicPage(name: "notifications", title: "Configure the way that Hubitat will get motion alert and ring events. Choose one of the following methods. IFTTT is highly preferred.", nextPage: "mainPage") { section { href "ifttt", title: "IFTTT Integration and Documentation for Motion and Ring Alerts" } section { href "pollingPage", title: "Configure Polling for Motion and Ring Alerts" } } } def ifttt() { setupDingables() List dingables = state.dingables?.collect { getChildDeviceInternal(it) }?.findAll { it != null } dynamicPage(name: "ifttt", title: 'Using IFTTT To Receive Motion and Ring Events') { section('About IFTTT') { paragraph("IFTTT is a service that provides interoperability between many cloud services. Ring has implemented IFTTT triggers for ring and motion events. Triggers allow actions to run. One of the actions that IFTTT supports are web service calls (Webhooks). The overall control flow will start with a motion event or ring event at your Ring device. The device will notify the Ring cloud of the event. The Ring cloud will notify the IFTTT cloud of the event. The IFTTT cloud will make a web service call to the Hubitat cloud. The Hubitat cloud will push that call to the hub locally. The app will process it and send it to the correct device.") paragraph("For the above to function correctly, an IFTTT applet must be configured for each event type for each device") } section('Prerequisites') { paragraph("""- An IFTTT account - The Ring service authorized to your IFTTT account (https://ifttt.com/ring) - A Ring device that supports motion and/or ring events in IFTTT - The above device authorized to IFTTT through the Ring IFTTT service - The ability to use the Webhooks actions on your IFTTT account (This appears to be configured by default for new IFTTT accounts.)""") } section('Steps to create IFTTT Applets') { paragraph("For each of the devices above create an applet like the screenshot at the bottom of this page. After you fulfill the prerequisites above perform the following steps to create an applet.") paragraph("""- Navigate to the applet create screen or follow this link. - For the trigger event or "+This" event choose/search for Ring and then pick one of the two (motion or ring) trigger actions - Select the device you want to configure events for and click "Create trigger" - For the action or "+That" event choose/search for "Webhooks" and pick the "Make a web request" action - Use the URL below from the "Webhooks URL" section in the URL field without any changes. - Choose "POST" for the Method. - Choose "application/json" for the Content Type. For the purposes of Hubitat and these notifications the Content Type field is NOT optional even though it says it is. - For the Body use the helpful copy and paste snippets created below. There should be one for each of the supported devices. If you chose ring for the trigger action choose the body payload for ring events. If you chose motion as the trigger action above choose the body payload for motion events. For the purposes of Hubitat and these notifications the Body field is NOT optional even though it says it is. - Click "Create action" and test the results.""") paragraph('You must visit https://ifttt.com to configure the applets.') } section('Webhooks URL') { paragraph("${getFullApiServerUrl()}/ifttt?access_token=${state.accessToken}") paragraph("""The URL breaks down into the following pieces: - "https://cloud.hubitat.com/api/" -- The base Hubitat cloud URL - The first GUID is your hub's unique ID. Don't disclose this to anybody. It's a little like your hub's username. - The numeric digits after the "/apps" portion is the installed app ID of this app. - The last GUID is the access token created by this app using OAuth that IFTTT will use to authenticate to Hubitat. DO NOT disclose this to anybody. It is like a password.""") } section('Body Payloads for Motion Events') { if (dingables) { paragraph(dingables.collect { "" + it.label + ": " + JsonOutput.toJson([kind: 'motion', motion: true, id: it.deviceNetworkId]) }.join('\n\n')) } else { paragraph("No installed devices support motion events") } } section('Body Payloads for Ring Events') { List ringables = dingables?.findAll { RINGABLES.contains(it.getDataValue("kind")) } if (ringables) { paragraph(ringables.collect { "" + it.label + ": " + JsonOutput.toJson([kind: 'ding', motion: false, id: it.deviceNetworkId]) }.join('\n\n')) } else { paragraph("No installed devices support ring events") } } section('Example Applet Screenshot') { //noinspection SpellCheckingInspection final String iftttScreenshotData = "" paragraph('
Screenshot
') } donationPageSection() } } def pollingPage() { configureDingPolling() dynamicPage(name: "pollingPage", title: "Configure Polling for Motion and Ring Alerts") { section('WARNING!! ADVERTENCIA!! ACHTUNG!! AVERTISSEMENT!!') { paragraph("Polling too quickly can have adverse affects on performance of your hubitat hub and may even get your Ring account temporarily or permanently locked. As of March 2022 Ring has taken no action to prevent polling.") paragraph("This is true for motion event, ring event, and light status polling.") paragraph("It is recommended to use the IFTTT method to receive notifcations instead of polling.") } section("Configure settings to poll for motions and rings:") { input name: "dingPolling", type: "bool", title: "Poll for motion and rings", defaultValue: false, submitOnChange: true if (dingPolling) { input name: "dingInterval", type: "number", range: "8..20", title: "Number of seconds in between motion/ring polls", defaultValue: 15, submitOnChange: true } } donationPageSection() } } def snapshots() { configureSnapshotPolling() dynamicPage(name: "snapshots", title: "Configure camera snapshot polling", nextPage: "mainPage") { section('

Snapshot Polling

') { paragraph("Snapshots provided by this app will are only available locally, and not via the cloud. If you access the dashboards with these images they will only display locally (when on the same network as the hub).") paragraph("Normally Ring only polls your devices for snapshots when an app on your account is open that needs image thumbnails (i.e. new thumbnails are only pulled you have the dashboard open on the phone app).") paragraph("""This has several side effects. """) } if (!state.dingables) { section('

Snapshot Prerequisite Error

') { paragraph('No installed devices support snapshots. Please install a device that supports snapshots before continuing') } return } section('

Configuration

') { input name: "snapshotPolling", type: "bool", title: "Poll for camera thumbnails", defaultValue: false, submitOnChange: true if (snapshotPolling == true) { input name: "snapshotInterval", type: "enum", title: "Interval between thumbnail refresh", required: true, options: snapshotIntervals, defaultValue: 120, submitOnChange: true } paragraph('

Device Configuration

') String msg = 'Click the links below and activate "Enable polling for thumbnail snapshots on this device" for each device you want to see. Refresh this page after enabling\n\n' for (final String dingable in state.dingables) { final String dingableDNI = getFormattedDNI(dingable) ChildDeviceWrapper child = getChildDevice(dingableDNI) if (child) { msg += """${child.label} (Opted ${state.snappables?.get(dingableDNI) ? 'in to' : 'out of'} snapshot polling)\n""" } } paragraph(msg) paragraph('

How to include snapshots on a dashboard

') paragraph("""""") paragraph('

Snapshot URLs

') if (!state.snappables) { paragraph("There are no cameras configured to poll for snapshots.") return } paragraph("WARNING: Do *not* share these URLs publicly") for (final String snappable in getEnabledSnappables()) { final URI url = new URI("${getFullLocalApiServerUrl()}/snapshot/${URLEncoder.encode(snappable, "UTF-8")}?access_token=${state.accessToken}") paragraph("""${getChildDeviceInternal(snappable).label}:\nURL: $url\nSnapshot""") } } donationPageSection() } } def deviceDiscovery() { if (!loggedIn()) { return dynamicPage(name: "deviceDiscovery", title: "Not authenticated", nextPage: "login", uninstall: true) { section("Please check your Ring username and password") { paragraph "" } } } logDebug "deviceDiscovery()" if (!selectedLocations) { return dynamicPage(name: "deviceDiscovery", title: "No locations selected!", nextPage: "locations", uninstall: true) { section("Please check your Ring location setup") { paragraph "" } } } List apiResponse = apiRequestDevices() if (!apiResponse) { return dynamicPage(name: "deviceDiscovery", title: "No devices found!", nextPage: "mainPage", uninstall: true) { section("apiRequestDevices returned nothing. See logs for more details") { paragraph "" } } } loadAvailableDevices(apiResponse) return dynamicPage(name: "deviceDiscovery", title: "Discovery Started!", nextPage: "addDevices", uninstall: true) { section("Select the devices you want created as Hubitat devices") { input "selectedDevices", "enum", title: "Select Ring Device(s)", multiple: true, options: getAvailableDevicesOptions() } } } void donationPageSection() { section { paragraph "
Donations greatly appreciated!
Paypal
" } } // Sets state.devices to a List of Maps of the form: {kind=base_station_v1, name=Ring Alarm Base Station - Alarm Base Station, id=dev_id_num} void loadAvailableDevices(List apiRequestDevicesResponse) { logDebug "loadAvailableDevices()" state.devices = [] for (final Map node in apiRequestDevicesResponse) { final String kind = node?.kind if (!settings.selectedLocations.contains(node?.location_id)) { logDebug "loadAvailableDevices: Excluding ${kind} at location ${node.location_id} because it is not in selected locations ${settings.selectedLocations}" } else if (!DEVICE_TYPES.containsKey(kind)) { logDebug "loadAvailableDevices: Excluding ${kind} at location ${node.location_id} because kind '${kind}' is not supported" } else { logDebug "loadAvailableDevices: Including a ${kind} at location ${node.location_id}" state.devices.add([name: "${DEVICE_TYPES[kind].name} - ${node.description}", id: node.id, kind: node.kind]) } } logTrace "loadAvailableDevices: supportedIds ${state.devices}" } // @return a Map like: [dev_id_num:device_name] Map getAvailableDevicesOptions() { logDebug "createAvailableDevicesOptions()" Map map = [:] for (final Map device in state.devices) { final String value = device.name final String key = device.id // @todo If no one reports this error, simplify this code if (map.containsKey(key)) { log.error("getAvailableDevicesOptions(): Device id '${key}' shows up more than once in apiRequestDevicesResponse. Please report this error so") } map[key] = map[key] ? map[key] + " || " + value : value } return map } void installed() { initialize() } void updated() { initialize() } void initialize() { logDebug "initialize()" configureDingPolling() configureSnapshotPolling() schedulePeriodicMaintenance() } mappings { path("/ifttt") { action: [POST: "processIFTTT"] } path("/snapshot/:ringDeviceId") { action: [GET: "serveSnapshot"] } } void processIFTTT() { def json try { json = parseJson(request.body) } catch (Exception e) { log.error "JSON received from IFTTT is invalid! ${request.body}: ${e}" return } final String kind = json.kind logDebug "processIFTTT() with ${kind} for ${json.id}" ChildDeviceWrapper d = getChildDeviceInternal(json.id) logTrace "data received: kind: ${kind}, id: ${json.id}, device: ${d}, request: ${request}, data: ${request.body}" if (!d) { log.warn "Received processIFTTT for device ${json.id} that does not exist" return } if (kind == "motion") { d.handleMotion(json) } else if (kind == "ding") { d.handleDing(json) } else { log.warn "Received processIFTTT for device ${json.id} with unsupported kind '${kind}'" } } def addDevices() { if (!loggedIn()) { return dynamicPage(name: "addDevices", title: "Not authenticated", nextPage: "login", uninstall: true) { section("Please check your Ring username and password") { paragraph "" } } } List devices = state.devices logTrace "devices ${devices}" ChildDeviceWrapper apiDevice = getAPIDevice() if (!apiDevice) { return dynamicPage(name: "addDevices", title: "Error", nextPage: "mainPage", uninstall: true) { section("getAPIDevice returned null. See logs for more details") { paragraph "" } } } String sectionText = "" boolean hubAdded = false Set enabledHubDoorbotIds = [].toSet() for (final String id in selectedDevices) { Map selectedDevice = devices.find { Map dev -> dev.id.toString() == id } logTrace "addDevices: Selected id ${id}, Selected device ${selectedDevice}" if (!selectedDevice) { final String tmpMsg = "addDevices: Error adding device id: '${id}'. Available devices: ${devices}" log.error(tmpMsg) sectionText += tmpMsg continue } final Integer selectedDeviceId = selectedDevice.id final String kind = selectedDevice.kind final Map deviceType = DEVICE_TYPES[kind] if (deviceType == null) { final String tmpMsg = "addDevices: Error adding device '${selectedDevice.name}'. Kind '${kind}' is not supported" log.error(tmpMsg) sectionText += tmpMsg continue } boolean isHubDevice = false ChildDeviceWrapper d = null if (selectedDevice) { d = getChildDeviceInternal(selectedDeviceId) } if (d == null) { logDebug selectedDevice if (["base_station_k1", "base_station_v1", "beams_bridge_v1"].contains(kind)) { isHubDevice = true enabledHubDoorbotIds.add(selectedDeviceId) if (!apiDevice.isHubPresent(selectedDeviceId)) { hubAdded = true sectionText += "Requesting API device to create ${selectedDevice.name}\r\n" } } else { try { log.warn "Creating a ${deviceType.name} with dni: ${getFormattedDNI(selectedDeviceId)}" d = addChildDevice("ring-hubitat-codahq", deviceType.driver, getFormattedDNI(selectedDeviceId), [label: selectedDevice?.name ?: deviceType.name]) d.refresh() sectionText += "Successfully added ${deviceType.name} with DNI ${getFormattedDNI(selectedDeviceId)}\r\n" } catch (e) { if (e.toString().replace(deviceType.driver, "") == "com.hubitat.app.exception.UnknownDeviceTypeException: Device type '' in namespace 'ring-hubitat-codahq' not found") { final String tmpMsg = "The '${deviceType.driver}' driver for device '${deviceType.name}' was not found and needs to be installed. NOTE: If you installed this using HPM, you can fix this by going to \"Update\" in HPM and selecting the optional drivers you need." log.error tmpMsg sectionText += '' + tmpMsg + '\r\n' } else { final String tmpMsg = "An error occured creating device: ${e}" log.error tmpMsg sectionText += tmpMsg + '\r\n' } } } } if (!isHubDevice) { d?.updateDataValue("kind", kind) d?.updateDataValue("kind_name", deviceType.name) } } apiDevice.setEnabledHubDoorbotIds(enabledHubDoorbotIds) if (hubAdded) { // Will add the hub zids to the API device's state. // If any selected hub devices have not been created, this will create them automatically apiDevice.updateTokensAndReconnectWebSocket() } return dynamicPage(name: "addDevices", title: "Devices Added", nextPage: "mainPage", uninstall: true) { if (sectionText) { logDebug sectionText section("Please Note!") { paragraph "Alarm base stations, Smart Lighting bridges and all devices connected to them are children of the API device.\r\n" } section("Add Ring Device Results:") { paragraph sectionText } } else { section("No devices added") { paragraph "All selected devices have previously been added" } } } } def uninstalled() { getChildDevices()?.each { deleteChildDevice(it.deviceNetworkId) } } void setupDingables() { state.dingables = getChildDevices()?.findAll { RINGABLES.contains(it.getDataValue("kind")) }?.collect { getRingDeviceId(it.deviceNetworkId) } } void configureDingPolling() { unschedule(getDings) if (dingPolling) { setupDingables() runIn(dingInterval ?: 15, getDings) } } void getDings() { apiRequestDings() if (dingPolling) { runIn(dingInterval ?: 15, getDings) } } // Get a random integer from [0..bound) Integer getRandomInteger(Integer bound) { return new Random().nextInt(bound) } void configureSnapshotPolling() { logDebug "configureSnapshotPolling()" unschedule(updateSnapshots) if (snapshotPolling) { final Integer interval = snapshotInterval?.toInteger() ?: 600 logInfo "Snapshot polling started with an interval of ${snapshotIntervals[interval].toLowerCase()}." setupDingables() //let's spread schedules out so that there is some randomness in how we hit the ring api final Integer currSec = getRandomInteger(60) final Integer altSec = (currSec + 30) > 59 ? currSec - 30 : currSec + 30 final Integer currMin = getRandomInteger(60) if (interval == 30) { final String secString = currSec > altSec ? "${altSec},${currSec}" : "${currSec},${altSec}" schedule("${secString} * * * * ? *", updateSnapshots) } else if (interval == 60) { schedule("${currSec} * * * * ? *", updateSnapshots) } else if (interval == 90) { final Integer startMin = currMin % 3 // Minute to start job final Integer offset = currSec >= 30 ? 1 : 0 schedule("${currSec} ${startMin}/3 * * * ? *", updateSnapshots) schedule("${altSec} ${(startMin + 1 + offset) % 3}/3 * * * ? *", updateSnapshots, [overwrite: false]) } else if (interval in 120..1800) { // Minutes final Integer mins = interval / 60 schedule("${currSec} ${currMin % mins}/${mins} * * * ? *", updateSnapshots) } else if (interval in 3600..43200) { // Hours final Integer hours = interval / 60 / 60 schedule("${currSec} ${currMin} 0/${hours} * * ? *", updateSnapshots) } else if (interval == 86400) { // Day schedule("${currSec} ${currMin} ${getRandomInteger(24)} * * ? *", updateSnapshots) } else { log.error("configureSnapshotPolling Unsupported interval ${interval}") } } } void updateSnapshots() { // This gets the snapshot timestamps only for selected timestamps, then retrieves those snapshots after a delay. // This is better than using snapshot-update, so battery powered devices can sleep if the user wants them to apiRequestSnapshotTimestamps(getEnabledSnappables()?.collect { final String it -> getRingDeviceId(it).toInteger() }) } // Kept for compatibility with old installs void prepSnapshots() { unschedule(prepSnapshots) unschedule(prepSnapshotsAlt) configureSnapshotPolling() } void prepSnapshotsAlt() { prepSnapshots() } void getSnapshots() { } // Don't need to do anything because this was only called with runIn def serveSnapshot() { final String ringDeviceId = URLDecoder.decode(params.ringDeviceId, "UTF-8") logDebug "serveSnapshot(${ringDeviceId})" if (!getChildDeviceInternal(ringDeviceId)) { log.error "serveSnapshot: Could not locate a device with an id of ${ringDeviceId}" return ["error": true, "type": "SmartAppException", "message": "Not Found"] } byte[] img = (byte[])state.snapshots[ringDeviceId]?.toArray() String strImg if (!img || img.length == 0) { logTrace "Default to missing image for ${ringDeviceId}" strImg = MISSING_IMG } else { strImg = 'data:image/png;base64,' + img.encodeBase64() } render contentType: "image/svg+xml", data: """""", status: 200 } @Field final static String MISSING_IMG = "" void snapshotOption(String cam, value) { if (!state.snappables) { state.snappables = [:] } state.snappables[cam] = value } Set getEnabledSnappables() { return state.snappables?.findAll { it.value }?.keySet() } @Field final static Map snapshotIntervals = [ 30: "30 Seconds", 60: "1 Minute", 90: "1.5 Minutes", 120: "2 Minutes", 180: "3 Minutes", 240: "4 Minutes", 300: "5 Minutes", 360: "6 Minutes", 600: "10 Minutes", 720: "12 Minutes", 900: "15 Minutes", 1200: "20 Minutes", 1800: "30 Minutes", 3600: "1 Hour", 7200: "2 Hours", 10800: "3 Hours", 14400: "4 Hours", 21600: "6 Hours", 28800: "8 Hours", 43200: "12 Hours", 86400: "24 Hours" ].asImmutable() // Resets the temporary access token. state.refresh_token can be used to get a net access token void resetAccessToken() { state.access_token = null state.access_token_expires = null } void addHeadersToHttpRequest(Map params, Map args = [:]) { params.headers = [ 'User-Agent': 'android:com.ringapp:3.25.0(26452333)', 'app_brand': 'ring', 'Accept-Encoding': 'gzip, deflate', 'Connection': 'Keep-Alive', ] params.headers['Host'] = new URI(params.uri).host if (args.hardware_id) { params.headers['Hardware_ID'] = state.appDeviceId } if (args.extra) { params.headers << args.extra } } void addAuthHeadersToHttpRequest(Map params) { params.headers['Authorization'] = "Bearer ${state.access_token}" } // For compatibility with old installs void authenticate() { authRefreshToken() } // Simple wrapper for apiRequestAuth* functions. Used to make mocking for tests simpler boolean authPassword() { return apiRequestAuthPassword() } boolean authRefreshToken() { return apiRequestAuthRefreshToken() } boolean authTwoFactorChallenge(final twoFactorCode) { return apiRequestAuthTwoFactorChallenge(twoFactorCode) } // Makes a ring api auth password request boolean apiRequestAuthPassword() { logTrace("apiRequestAuthPassword()") resetAccessToken() state.refresh_token = null state.session_token = null // Generate an ID so that Ring doesn't think these are all coming from the same device state.appDeviceId = UUID.randomUUID().toString() state.authResponse = "apiRequestAuthPassword: This shouldn't happen" Map params = makeAuthParams([ grant_type: 'password', password: settings.password, username: settings.username ]) return apiRequestAuthCommon("apiRequestAuthPassword", params) } // Makes a ring api auth refresh token request boolean apiRequestAuthRefreshToken() { logTrace("apiRequestAuthRefreshToken()") state.authResponse = "apiRequestAuthRefreshToken: This shouldn't happen" if (!state.refresh_token) { log.error 'apiRequestAuthRefreshToken: Refresh token is not valid. Unable to authenticate with Ring servers.' state.authResponse = "apiRequestAuthRefreshToken: Refresh token is not valid. Unable to authenticate with Ring servers." return false } Map params = makeAuthParams([ grant_type: 'refresh_token', refresh_token: state.refresh_token ]) addAuthHeadersToHttpRequest(params) return apiRequestAuthCommon("apiRequestAuthRefreshToken", params) } // Makes a ring api auth two factor challenge request boolean apiRequestAuthTwoFactorChallenge(final twoFactorCode) { logTrace("apiRequestAuthTwoFactorChallenge(${twoFactorCode})") state.authResponse = "apiRequestAuthTwoFactorChallenge: This shouldn't happen" Map params = makeAuthParams([ grant_type: 'password', password: settings.password, username: settings.username ]) params.headers << ['2fa-support': 'true', '2fa-code': twoFactorCode] return apiRequestAuthCommon("apiRequestAuthTwoFactorChallenge", params) } boolean apiRequestAuthCommon(final String funcName, final Map params) { try { boolean success = false httpPost(params) { resp -> def body = resp.data logInfo "$funcName succeeded" logTrace "$funcName succeeded, body: ${body}" state.access_token = body.access_token state.refresh_token = body.refresh_token logDebug "access token: ${state.access_token}, refresh token: ${state.refresh_token}" success = state.access_token && state.refresh_token state.authResponse = "success" if (success) { state.holdRequests = false if (body.expires_in?.toString()?.isInteger()) { int interval = body.expires_in.toInteger() state.access_token_expires = now() + (interval - 20) * 1000 logInfo "OAuth token expires in ${interval} seconds. Scheduling refresh in ${interval - 20} seconds." runIn(interval - 20, apiRequestAuthRefreshToken) } } } return success } catch (groovyx.net.http.HttpResponseException ex) { final Integer status = ex.getStatusCode() final resp = ex.getResponse() final body = resp.getData() if (status == 400) { if (body instanceof Map) { if (body.error?.contains('Verification Code')) { state.authResponse = resp.getData().error } else { state.authResponse = "$funcName HTTP ${status} error: ${JsonOutput.toJson(body)}" } } else { state.authResponse = "$funcName HTTP ${status} error: ${body}" } } else if (status == 412) { logInfo "2 Step Challenge" state.holdRequests = true state.authResponse = "challenge" return false } else if (status == 429) { state.holdRequests = true state.authResponse = "$funcName HTTP 429 error. Sending too many requests. Try again later" } else { logDebug "$funcName HTTP ${status} error. Exception: ${ex}. ${body}" if (body instanceof Map) { String errorDescription = body?.error_description if (errorDescription != null) { if (errorDescription == 'too many requests from dependency service') { state.authResponse = 'You have requested too many 2fa codes. Ring limits 2fa to 10 codes within 10 minutes. Please try again in 10 minutes.' } else if (errorDescription == "invalid user credentials") { state.authResponse = 'Invalid username/password' } else { state.authResponse = "$funcName HTTP ${status} error: ${errorDescription}" } } else { state.authResponse = "$funcName HTTP ${status} error: ${JsonOutput.toJson(body)}" } } else { state.authResponse = "$funcName HTTP ${status} error: ${body}" } } log.warn(state.authResponse) } catch (ConnectTimeoutException | HttpHostConnectException | SSLPeerUnverifiedException | SSLHandshakeException | \ SocketException | SocketTimeoutException | NoRouteToHostException | UnknownHostException | ResponseParseException e) { state.authResponse = "$funcName. Authentication failed because of a transient error. ${e}" log.warn(state.authResponse) } return false } Map makeAuthParams(final Map grantData) { Map params = [ uri: 'https://oauth.ring.com/oauth/token', contentType: JSON, body: grantData + [ "client_id": "ring_official_android", "scope": "client" ], ] addHeadersToHttpRequest(params, [hardware_id: true, extra: [ "User-Agent" : "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/75.0.3770.100 Safari/537.36", ]]) return params } boolean authTokenOk() { final Long timeBeforeExpiry = ((state.access_token_expires ?: now()) - now()) / 1000 if (timeBeforeExpiry < 2) { logDebug "Auth token is about to expire" return false } return true } /** * This gets called when we get a 404 error complaining the Client Device can't be found. Why this is still necessary * isn't entirely clear. The errors only happen occasionally. */ boolean apiRequestAuthSession() { logTrace("apiRequestAuthSession()") Map params = makeClientsApiParams('/session', [ requestContentType: JSON, body: [ device: [ hardware_id: state.appDeviceId, metadata: [api_version: 11], os: 'android', // can use android, ios, ring-site, windows for sure ], ] ]) return apiRequestSyncCommon("apiRequestAuthSession", false, params) { Map reqParams -> boolean retval = false httpPost(reqParams) { resp -> def body = resp.data logTrace "apiRequestAuthSession succeeded, body: ${JsonOutput.toJson(body)}" state.session_token = body.profile.authentication_token logDebug "apiRequestAuthSession succeeded, token is ${state.session_token}" retval = true } return retval } } // Makes a ring api request for location data List apiRequestDevices() { logTrace("apiRequestDevices()") Map params = makeClientsApiParams('/ring_devices', [query: [api_version: 11]]) return apiRequestSyncCommon("apiRequestDevices", false, params) { Map reqParams -> List retval = null httpGet(reqParams) { resp -> def body = resp.data logTrace "apiRequestDevices succeeded, body: ${JsonOutput.toJson(body)}" // @note Intenionally leaving out "beams" because they are handled by the beams bridge device retval = [] for (final String key in ['authorized_doorbots', 'base_stations', 'beams_bridges', 'chimes', 'doorbots', 'stickup_cams']) { retval += body[key] } } return retval } } /** * Makes a ring api request for dings data * @param dni deviceNetworkId of device to refresh * @todo Keys that could be useful: * settings.motion_detection_enabled [true/false] // set by modes or Record Motion toggle * settings.power_mode ['battery'/'wired'] // some battery cams can be wired and set to operate in 'wired' mode * alerts.connection ['online'/'offline'] * alerts.battery: ['low'] */ void apiRequestDeviceRefresh(final String dni) { logTrace("apiRequestDeviceRefresh(${dni})") Map params = makeClientsApiParams('/ring_devices/' + getRingDeviceId(dni), [query: [api_version: 11]]) apiRequestAsyncCommon("apiRequestDeviceRefresh", "Get", params, false) { resp -> def body = resp.getJson() logTrace "apiRequestDeviceRefresh for ${dni} succeeded, body: ${JsonOutput.toJson(body)}" ChildDeviceWrapper d = getChildDevice(dni) if (d) { d.handleRefresh(body) } else { log.error "apiRequestDeviceRefresh cannot get child device with dni ${dni}" } } } /** * Makes a ring api request to control a device * @param dni DNI of device to refresh * @param kind Kind of device ("chimes", etc.) * @param action Action to perform on device ("floodlight_light_off", etc) * @param query Action to perform on device ([kind: "motion"], etc) */ void apiRequestDeviceControl(final String dni, final String kind, final String action, final Map query) { logTrace("apiRequestDeviceControl(${dni}, ${kind}, ${action}, ${query})") Map params = makeClientsApiParams('/' + kind + '/' + getRingDeviceId(dni) + '/' + action, [contentType: TEXT, requestContentType: JSON, query: query]) apiRequestAsyncCommon("apiRequestDeviceControl", "Post", params, false) { resp -> logTrace "apiRequestDeviceControl ${kind} ${action} for ${dni} succeeded" def body = resp.getData() ? resp.getJson() : null ChildDeviceWrapper d = getChildDevice(dni) if (d) { d.handleDeviceControl(action, body, query) } else { log.error "apiRequestDeviceControl ${kind}.${action} (${query}) cannot get child device with dni ${dni}" } } } /** * Makes a ring api request to set a value for a device * @param dni DNI of device to refresh * @param kind Kind of device ("doorbots", etc.) * @param action Action to perform on device ("floodlight_light_off", etc) */ void apiRequestDeviceSet(final String dni, final String kind, final String action = null, final Map query = null) { logTrace("apiRequestDeviceSet(${dni}, ${kind}, ${action}, ${query})") Map params = makeClientsApiParams('/' + kind + '/' + getRingDeviceId(dni) + (action ? "/${action}" : ""), [contentType: TEXT, requestContentType: JSON, query: query]) apiRequestAsyncCommon("apiRequestDeviceSet", "Put", params, false) { resp -> logTrace "apiRequestDeviceSet ${kind} ${action} for ${dni} succeeded" def body = resp.getData() ? resp.getJson() : null ChildDeviceWrapper d = getChildDevice(dni) if (d) { d.handleDeviceSet(action, body, query) } else { log.error "apiRequestDeviceSet ${kind}.${action} cannot get child device with dni ${dni}" } } } /** * Makes a ring api request to set a value for a device * @param dni DNI of device to refresh * @param kind Kind of device ("doorbots", "chimes", etc.) */ void apiRequestDeviceHealth(final String dni, final String kind) { logTrace("apiRequestDeviceHealth(${dni}, ${kind})") Map params = makeClientsApiParams('/' + kind + '/' + getRingDeviceId(dni) + '/health', [contentType: TEXT, requestContentType: JSON]) apiRequestAsyncCommon("apiRequestDeviceHealth", "Get", params, false) { resp -> def body = resp.getData() ? resp.getJson() : null logTrace "apiRequestDeviceHealth ${kind} for ${dni} succeeded, body: ${JsonOutput.toJson(body)}" ChildDeviceWrapper d = getChildDevice(dni) if (d) { d.handleHealth(body) } else { log.error "apiRequestDeviceHealth ${kind} cannot get child device with dni ${dni}" } } } /** * Makes a ring api request for dings data * * @todo The returned object has a detection_type. There's also a settings.cv.settings.detection_types in the apiRequestDevices response */ void apiRequestDings() { logTrace("apiRequestDings()") Map params = makeClientsApiParams('/dings/active', [query: [api_version: 11]]) apiRequestAsyncCommon("apiRequestDings", "Get", params, false) { resp -> def body = resp.getJson() logTrace "apiRequestDings succeeded, body: ${JsonOutput.toJson(body)}" for (final Map dingInfo in body) { final String deviceId = dingInfo.doorbot_id if (state.dingables.contains(deviceId)) { logTrace "apiRequestDings: Got ding for ${getFormattedDNI(deviceId)}" ChildDeviceWrapper d = getChildDeviceInternal(deviceId) if (d) { if (dingInfo.kind == "motion") { d.handleMotion(dingInfo) } else if (dingInfo.kind == "ding") { d.handleDing(dingInfo) } else { log.warn "apiRequestDings: Received unsupported kind '${dingInfo.kind}' for device ${deviceId}" } } else { log.error "apiRequestDings Received ding for device '${deviceId}' that does not exist" } } } } } // Makes a ring api request for location data List apiRequestLocations() { logTrace("apiRequestLocations()") Map params = [ uri: DEVICES_API_BASE_URL + '/locations', contentType: JSON ] addHeadersToHttpRequest(params) return apiRequestSyncCommon("apiRequestLocations", false, params) { Map reqParams -> List retval = null httpGet(reqParams) { resp -> def body = resp.getData() logTrace "apiRequestLocations succeeded, body: ${JsonOutput.toJson(body)}" retval = body.user_locations } return retval } } /** * Makes a ring api request for location data * @param dni DNI of device to pass mode to * @warning Only works if there is no alarm present */ void apiRequestModeGet(final String dni) { logTrace("apiRequestModeGet(${dni})") Map params = makeAppApiParams("/mode/location/${getSelectedLocation()?.id}", [query: [api_version: 11]], [hardware_id: true]) apiRequestAsyncCommon("apiRequestModeGet", "Get", params, false) { resp -> def body = resp.getData() ? resp.getJson() : null logTrace "apiRequestModeGet succeeded, body: ${JsonOutput.toJson(body)}" ChildDeviceWrapper d = getChildDevice(dni) if (d) { d.updateMode(body.mode) } else { log.error "apiRequestModeGet cannot get child device with dni ${dni}" } } } /** * Makes a ring api request to set mode value * @param dni DNI of device to set mode on * @param mode Mode to set * @warning Only works if there is no alarm present */ void apiRequestModeSet(final String dni, final String mode) { logTrace("apiRequestModeSet(${dni}, ${mode})") Map params = makeAppApiParams("/mode/location/${getSelectedLocation()?.id}", [body: [mode: mode, readOnly: true], query: [api_version: 11]], [hardware_id: true]) apiRequestAsyncCommon("apiRequestModeSet", "Post", params, false) { resp -> def body = resp.getData() ? resp.getJson() : null logTrace "apiRequestModeSet for mode ${mode} succeeded, body: ${JsonOutput.toJson(body)}" ChildDeviceWrapper d = getChildDevice(dni) if (d) { d.updateMode(body.mode) } else { log.error "apiRequestModeSet cannot get child device with dni ${dni}" } } } /** * Makes a ring api request for a new websocket ticket */ void apiRequestTickets(final String dni) { logTrace("apiRequestTickets(${dni})") Map params = makeAppApiParams("/clap/tickets", [requestContentType: TEXT, query: [locationID: selectedLocations.toString()]]) addHeadersToHttpRequest(params) apiRequestAsyncCommon("apiRequestTickets", "Get", params, false) { resp -> def body = resp.getData() ? resp.getJson() : null logTrace "apiRequestTickets succeeded, body: ${JsonOutput.toJson(body)}" ChildDeviceWrapper d = getChildDevice(dni) if (d) { d.updateTickets(body) } else { log.error "apiRequestTickets cannot get child device with dni ${dni}" } } } /** * Makes a ring api request for location data * @param dni DNI of device to request image for */ void apiRequestSnapshotImages(final Map data) { logTrace("apiRequestSnapshotImages(${data})") if (!state.snapshots) { state.snapshots = [:] } for (final doorbotId in data.doorbotIds) { final localDoorbotId = doorbotId.toInteger() Map params = [uri: CLIENTS_API_BASE_URL + '/snapshots/image/' + localDoorbotId, requestContentType: JSON] addHeadersToHttpRequest(params, [hardware_id: true, extra: ["Accept": "application.vnd.api.v11+json"]]) // Would like to do this async, but for some reason the state won't get updated when async apiRequestSyncCommon("apiRequestSnapshotImages", false, params) { Map reqParams -> httpGet(reqParams) { resp -> logTrace "apiRequestSnapshotImages succeeded for ${localDoorbotId}" if (resp.getData()) { byte[] retval = new byte[resp.data.available()] resp.data.read(retval) state.snapshots[getFormattedDNI(localDoorbotId)] = retval } } } } } /** * Makes a ring api request for snapshot timestamps * @param doorbotIds List of snappable doorbot ids to get timestamps for */ void apiRequestSnapshotTimestamps(List doorbotIds) { logTrace("apiRequestSnapshotTimestamps(${doorbotIds})") Map params = makeClientsApiParams('/snapshots/timestamps', [body: [doorbot_ids: doorbotIds], requestContentType: JSON], [hardware_id: true, extra: ["Accept": "application.vnd.api.v11+json"]]) // Would like to do this async, but for some reason the state won't get updated when async apiRequestSyncCommon("apiRequestSnapshotTimestamps", false, params) { Map reqParams -> httpPost(reqParams) { resp -> def body = resp.getData() logTrace "apiRequestSnapshotTimestamps for ${doorbotIds} succeeded, body: ${JsonOutput.toJson(body)}" // @todo Consider comparing new timestamps to old timestamps. Could use this to avoid getting a snapshot when there is no update state.lastSnapshotTimestamps = body.timestamps final List returnedDoorbotIds = body.timestamps*.doorbot_id logTrace "apiRequestSnapshotTimestamps returned these doorbots: ${returnedDoorbotIds}" final Set nonReturnedDoorbotIds = doorbotIds.toSet() - returnedDoorbotIds.toSet() if (nonReturnedDoorbotIds) { log.warn("apiRequestSnapshotTimestamps returned fewer doorbot ids than requested. Missing doorbots: ${nonReturnedDoorbotIds}") } runIn(15, apiRequestSnapshotImages, [data: [doorbotIds: returnedDoorbotIds]]) } } } /** * @param functionName Name of the apiRequest being run * @param reauthCall Set to true if this is a retry after attempting to reauth * @param httpRequestClosure Closure that executes the api request and returns a value * @return Value from httpRequestClosure, or null on error */ Object apiRequestSyncCommon(final String functionName, boolean reauthCall, Map params, Closure httpRequestClosure) { if (!authTokenOk()) { logDebug("Refreshing auth token before running ${functionName}") authRefreshToken() } addAuthHeadersToHttpRequest(params) try { return httpRequestClosure.call(params) } catch (groovyx.net.http.HttpResponseException ex) { final Integer status = ex.getStatusCode() final resp = ex.getResponse() if (isClientDeviceNotFoundHttpError(resp.getData(), status, params.uri, functionName)) { if (apiRequestAuthSession()) { return apiRequestSyncCommonRetry(functionName, params, httpRequestClosure) } } else if (status == 401) { logDebug "$functionName HTTP 401" resetAccessToken() // If server hasn't said we're sending too many requests, try refreshing auth code if (!reauthCall && !state.holdRequests && authRefreshToken()) { return apiRequestSyncCommonRetry(functionName, params, httpRequestClosure) } } else if (status == 429) { log.error "$functionName HTTP 429. Sending too many requests. Retrying in 200 seconds. Report this error so proper handling can be added" log.debug "$functionName HTTP 429 headers: ${resp.getAllHeaders()}" resetAccessToken() state.holdRequests = true return null } log.error "$functionName HTTP error. Exception: ${ex.getMessage()}. ${resp.getData()}" } catch (ConnectTimeoutException | HttpHostConnectException | SSLPeerUnverifiedException | SSLHandshakeException | \ SocketException | SocketTimeoutException | NoRouteToHostException | UnknownHostException | ResponseParseException e) { log.warn "${functionName}. Failed because of a transient error. ${e}" } catch (Exception e) { log.error "${functionName}. Caught an unhandled exception. ${e}" } return null } // This function just calls apiRequestSyncCommon with reauthCall=true. This function is only here to make testing a bit easier Object apiRequestSyncCommonRetry(final String functionName, Map params, Closure httpRequestClosure) { return apiRequestSyncCommon(functionName, true, params, httpRequestClosure) } /** * @param functionName Name of the apiRequest being run * @param httpFunction Name of async http function to call * @param params Parameters to pass to httpFunction * @param reauthCall Set to true if this is a retry after attempting to reauth * @param successClosure Closure that executes the api request and returns a value * @param retryCount Number of times this request has been retried. */ void apiRequestAsyncCommon(final String functionName, final String httpFunction, final Map params, boolean reauthCall, Closure successClosure, Integer retryCount = 0) { if (!authTokenOk()) { logInfo("Refreshing auth token before running ${functionName}") authRefreshToken() } addAuthHeadersToHttpRequest(params) try { "asynchttp${httpFunction}"(apiRequestAsyncCallbackCommon, params, [ functionName: functionName, httpFunction: httpFunction, params: params, reauthCall: reauthCall, retryCount: retryCount, successClosure: successClosure, ]) } catch (final Exception e) { log.error "${functionName} - General Exception: ${e}" } } /** * @param data Required keys: functionName, httpFunction, params, reauthCall, retryCount, successClosure */ void apiRequestAsyncCallbackCommon(resp, Map data) { final String functionName = data.functionName logTrace(functionName) if (!resp.hasError()) { data.successClosure.call(resp) return } final Integer status = resp.getStatus() final String message = resp.getErrorMessage() if (isRetryableHttpError(status, message)) { String msg = functionName + " (retryable) http " if (isSSLHttpError(status, message)) { msg += "SSL error: " + message } else if (isDnsHttpError(status, message)) { msg += "DNS error: " + message } else { msg += "error: " + status + ", " + message + ", " + resp.errorData } log.warn msg data.retryCount += 1 if (data.retryCount > 10) { log.error "${functionName}. Retries exceeded" } else { apiRequestAsyncCommonRetry(30, data) // Retry in 30 seconds } return } if (isClientDeviceNotFoundHttpError(resp.errorData, status, data.params.uri, functionName)) { if (apiRequestAuthSession()) { apiRequestAsyncCommon(functionName, data.httpFunction, data.params, false, data.successClosure, data.retryCount) return } } else if (status == 401) { logDebug "$functionName HTTP 401" resetAccessToken() // If server hasn't said we're sending too many requests, try refreshing auth code if (!data.reauthCall && !state.holdRequests && authRefreshToken()) { apiRequestAsyncCommon(functionName, data.httpFunction, data.params, true, data.successClosure, data.retryCount) return } } else if (status == 429) { log.error "$functionName HTTP 429. Sending too many requests. Retrying in 200 seconds. Report this error so proper handling can be added" log.debug "$functionName HTTP 429 headers: ${resp.getHeaders()}" resetAccessToken() state.holdRequests = true apiRequestAsyncCommonRetry(200, data) // Retry in 200 seconds return } log.error "${functionName} HTTP error. Status: ${status}, Error: ${message}. Body: ${resp.errorData}" } void apiRequestAsyncCommonRetry(Long delayInSeconds, Map data) { // @note I would really prefer to use runIn for the delay, but for some reason runIn executes the successClosure pauseExecution(delayInSeconds * 1000) apiRequestAsyncCommon(data.functionName, data.httpFunction, data.params, false, data.successClosure, data.retryCount) } /** * Sometimes Ring returns a 404 error saying the Client Device id isn't valid. An apiRequestAuthSession seems to fix this * Only seems to happen to CLIENTS_API_BASE_URL requests * * @param status Status code of the original request * @param uri URI of the request with an error * @param functionName Name of the function that made the original request * * @return True on match, false otherwise */ boolean isClientDeviceNotFoundHttpError(errorData, final Integer status, final String uri, final String functionName) { if (status == 404) { // Wrap this in a try/catch just in case the errorData isn't parsable json try { Map errorJson = errorData instanceof Map ? errorData : parseJson(errorData) if (uri?.startsWith(CLIENTS_API_BASE_URL) && errorJson?.error == "Client Device with ${state.appDeviceId} not found") { logDebug("$functionName Got a 404 error '${errorJson.error}'. Retrying request after a apiRequestAuthSession") return true } /* groovylint-disable-next-line EmptyCatchBlock */ } catch (Exception e) { } } return false } boolean isRetryableHttpError(final Integer status, final String message) { return (status == null && (message?.contains('timed out') || message?.contains('timeout'))) || (status == 408 || status > 500) } boolean isSSLHttpError(final Integer status, final String message) { return status == 408 && (message.contains("sun.security.validator.ValidatorException") || message.contains("sun.security.provider.certpath.SunCertPathBuilderException") || message.contains("java.security.cert.CertPathValidatorException")) } boolean isDnsHttpError(final Integer status, final String message) { // Equivalent to UnknownHostException and NoRouteToHostException return status == 408 && (message.contains("Name or service not known") || message.contains("No route to host")) } Map makeAppApiParams(final String urlSuffix, final Map args, final Map headerArgs = [:]) { Map params = [ uri: APP_API_BASE_URL + urlSuffix, contentType: args.getOrDefault('contentType', JSON), requestContentType: args.getOrDefault('requestContentType', JSON), ] params << args.subMap(['body', 'query']) addHeadersToHttpRequest(params, headerArgs) return params } Map makeClientsApiParams(final String urlSuffix, final Map args, final Map headerArgs = [:]) { Map params = [ uri: CLIENTS_API_BASE_URL + urlSuffix, contentType: args.getOrDefault('contentType', JSON), ] params << args.subMap(['body', 'requestContentType', 'query']) addHeadersToHttpRequest(params, headerArgs) return params } // Called by initialize and by child ring-api-virtual-device when an old version of things was detected void schedulePeriodicMaintenance() { schedule("0 ${getRandomInteger(60)} ${getRandomInteger(5)} ? * MON", periodicMaintenance) } void periodicMaintenance() { logInfo "Running periodic maintenance" // Clean up some things from old versions app.removeSetting('twofactor') state.remove('currentDisplayName') state.remove('currentDeviceId') // Remove entries from state.snappables that use the old 'RING-' naming for (final entry in state.snappables?.clone()) { if (entry.key.startsWith('RING-')) { final String correctDNI = getFormattedDNI(getRingDeviceId(entry.key)) // If an existing key with the correct name isn't found, move the value to the correct key. Otherwise assume // that the value with the correct style name is correct if (!state.snappables.containsKey(correctDNI)) { state.snappables[correctDNI] = entry.value } state.snappables.remove(entry.key) // Remove bad key } } for (final setting in settings) { if (setting.key.endsWith('_label')) { app.removeSetting(setting.key) } } getChildDevices()*.runCleanup() // Refresh cameras, lights, etc. This gets new firmware versions, updates wifi rssi, etc. for (final dingable in state.dingables) { getChildDeviceInternal(dingable)?.refresh() } } ChildDeviceWrapper getChildDeviceInternal(id) { if (id instanceof String && id.startsWith("RING")) { id = getRingDeviceId(id) } final String dni = getFormattedDNI(id) ChildDeviceWrapper d = getChildDevice(dni) if (!d) { d = getChildDevice('RING-' + id) if (d) { d.deviceNetworkId = dni log.warn "Migrated existing device RING-${id} to new DNI ${dni}" } } return d } ChildDeviceWrapper getAPIDevice(Map location = null) { if (location == null) { location = getSelectedLocation() } if (location == null) { log.error("getAPIDevice: No location defined") return null } final String formattedDNI = RING_API_DNI + "||" + location.id ChildDeviceWrapper d = getChildDevice(formattedDNI) if (!d) { d = getChildDevice("RING-" + RING_API_DNI) // Migrate if it's the old DNI if (d) { d.deviceNetworkId = formattedDNI log.warn "Migrated existing API device RING-${RING_API_DNI} to new DNI ${formattedDNI}" } // Create otherwise else { d = addChildDevice("ring-hubitat-codahq", "Ring API Virtual Device", formattedDNI, null, [label: location.name + " Location"]) logInfo "Ring API Virtual Device with ID ${formattedDNI} created" } d.updateDataValue("kind", RING_API_DNI) d.updateDataValue("kind_name", "Ring API Virtual Device") } return d } boolean loggedIn() { logDebug "loggedIn()" logTrace "state.access_token ${state.access_token}" return state.access_token != null } // @todo Refactor this. The naming here is a mess Map getSelectedLocation() { def loc = state.locationOptions.find { location -> selectedLocations.contains(location.key) || selectedLocations == location.key } return loc ? [id: loc.key, name: loc.value] : null } void logInfo(msg) { if (descriptionTextEnable) { log.info msg } } void logDebug(msg) { if (logEnable) { log.debug msg } } void logTrace(msg) { if (traceLogEnable) { log.trace msg } } String getFormattedDNI(final id) { return 'RING||' + id } String getRingDeviceId(String id) { if (id?.startsWith("RING||")) { return id.substring(6) } else if (id?.startsWith("RING-")) { return id.substring(5) } return id } @Field final static String CLIENTS_API_BASE_URL = 'https://api.ring.com/clients_api' @Field final static String DEVICES_API_BASE_URL = 'https://api.ring.com/devices/v1' @Field final static String APP_API_BASE_URL = 'https://app.ring.com/api/v1' @Field final static String RING_API_DNI = "WS_API_DNI" @Field final static String JSON = 'application/json' @Field final static String TEXT = 'text/plain' @Field final static Set RINGABLES = [ "doorbell", "doorbell_v3", "doorbell_v4", "doorbell_v5", "doorbell_graham_cracker", "doorbell_portal", "doorbell_scallop", "doorbell_scallop_lite", "doorbell_oyster", "cocoa_doorbell", "cocoa_doorbell_v2", "cocoa_floodlight", "lpd_v1", "lpd_v2", "lpd_v3", "lpd_v4", "jbox_v1", ].toSet().asImmutable() @Field final static Set DINGABLES = [ "cocoa_camera", "cocoa_doorbell", "cocoa_doorbell_v2", "cocoa_floodlight", "cocoa_spotlight", "doorbell_graham_cracker", "doorbell_portal", "doorbell_oyster", "doorbell_scallop_lite", "doorbell_scallop", "doorbell_v3", "doorbell_v4", "doorbell_v5", "doorbell", "floodlight_pro", "floodlight_v2", "hp_cam_v1", "hp_cam_v2", "jbox_v1", "lpd_v1", "lpd_v2", "lpd_v3", "lpd_v4", "spotlightw_v2", "stickup_cam_wired", "stickup_cam_elite", "stickup_cam_lunar", "stickup_cam_mini", "stickup_cam_v3", "stickup_cam_v4", "stickup_cam", ].toSet().asImmutable() @Field final static Map DEVICE_TYPES = [ "base_station_k1": [name: "Ring Alarm Base Station", driver: "SHOULD NOT SEE THIS"], "base_station_v1": [name: "Ring Alarm Base Station", driver: "SHOULD NOT SEE THIS"], "beams_bridge_v1": [name: "Ring Bridge Hub", driver: "SHOULD NOT SEE THIS"], "chime_pro_v2": [name: "Ring Chime Pro (v2)", driver: "Ring Virtual Chime"], "chime_pro": [name: "Ring Chime Pro", driver: "Ring Virtual Chime"], "chime_v2": [name: "Ring Chime V2", driver: "Ring Virtual Chime"], "chime": [name: "Ring Chime", driver: "Ring Virtual Chime"], "cocoa_camera": [name: "Ring Stick Up Cam", driver: "Ring Virtual Camera with Siren"], "cocoa_doorbell": [name: "Ring Video Doorbell 2020", driver: "Ring Virtual Camera"], "cocoa_doorbell_v2": [name: "Ring Video Doorbell 2023", driver: "Ring Virtual Camera"], "cocoa_floodlight": [name: "Ring Floodlight Cam Wired Plus", driver: "Ring Virtual Light with Siren"], "cocoa_spotlight": [name: "Ring Spotlight Cam Plus", driver: "Ring Virtual Light with Siren", dingable: true], "doorbell_graham_cracker": [name: "Ring Video Doorbell Wired", driver: "Ring Virtual Camera"], "doorbell_portal": [name: "Ring Peephole Cam", driver: "Ring Virtual Camera"], "doorbell_oyster": [name: "Ring Video Doorbell 4", driver: "Ring Virtual Camera"], "doorbell_scallop_lite": [name: "Ring Video Doorbell 3", driver: "Ring Virtual Camera"], "doorbell_scallop": [name: "Ring Video Doorbell 3 Plus", driver: "Ring Virtual Camera"], "doorbell_v3": [name: "Ring Video Doorbell", driver: "Ring Virtual Camera"], "doorbell_v4": [name: "Ring Video Doorbell 2", driver: "Ring Virtual Camera"], "doorbell_v5": [name: "Ring Video Doorbell 2", driver: "Ring Virtual Camera"], "doorbell": [name: "Ring Video Doorbell", driver: "Ring Virtual Camera"], "floodlight_pro": [name: "Ring Floodlight Cam Wired Pro", driver: "Ring Virtual Light with Siren"], "floodlight_v2": [name: "Ring Floodlight Cam Wired", driver: "Ring Virtual Light with Siren"], "hp_cam_v1": [name: "Ring Floodlight Cam", driver: "Ring Virtual Light with Siren"], "hp_cam_v2": [name: "Ring Spotlight Cam Wired", driver: "Ring Virtual Light with Siren"], "jbox_v1": [name: "Ring Video Doorbell Elite", driver: "Ring Virtual Camera"], "lpd_v1": [name: "Ring Video Doorbell Pro", driver: "Ring Virtual Camera"], "lpd_v2": [name: "Ring Video Doorbell Pro", driver: "Ring Virtual Camera"], "lpd_v3": [name: "Ring Video Doorbell Pro", driver: "Ring Virtual Camera"], "lpd_v4": [name: "Ring Video Doorbell Pro 2", driver: "Ring Virtual Camera"], "spotlightw_v2": [name: "Ring Spotlight Cam Wired", driver: "Ring Virtual Light with Siren"], "stickup_cam_wired": [name: "Ring Stick Up Cam Elite (2nd gen)", driver: "Ring Virtual Camera with Siren"], "stickup_cam_elite": [name: "Ring Stick Up Cam Elite (2nd gen)", driver: "Ring Virtual Camera with Siren"], "stickup_cam_lunar": [name: "Ring Stick Up Cam Battery/Solar (2nd gen)", driver: "Ring Virtual Camera with Siren"], "stickup_cam_mini": [name: "Ring Indoor Cam", driver: "Ring Virtual Camera with Siren"], "stickup_cam_v3": [name: "Ring Stick Up Cam (1st gen)", driver: "Ring Virtual Camera"], "stickup_cam_v4": [name: "Ring Spotlight Cam Battery/Solar", driver: "Ring Virtual Light"], "stickup_cam": [name: "Ring Stick Up Cam (1st gen)", driver: "Ring Virtual Camera"], ].asImmutable()