From 46039efd0a70b15d52cc1b68c7f0f7983459ea04 Mon Sep 17 00:00:00 2001 From: morph166955 <53797132+morph166955@users.noreply.github.com> Date: Mon, 26 Jun 2023 01:49:42 -0500 Subject: [PATCH] [androidtv] AndroidTV Binding initial contribution (#14282) Signed-off-by: Ben Rosenblum --- CODEOWNERS | 1 + bom/openhab-addons/pom.xml | 5 + bundles/org.openhab.binding.androidtv/NOTICE | 13 + .../org.openhab.binding.androidtv/README.md | 484 ++++++ bundles/org.openhab.binding.androidtv/pom.xml | 32 + .../src/main/feature/feature.xml | 10 + .../internal/AndroidTVBindingConstants.java | 56 + ...idTVDynamicCommandDescriptionProvider.java | 43 + .../androidtv/internal/AndroidTVHandler.java | 261 ++++ .../internal/AndroidTVHandlerFactory.java | 61 + .../GoogleTVDiscoveryParticipant.java | 101 ++ .../ShieldTVDiscoveryParticipant.java | 103 ++ .../protocol/googletv/GoogleTVCommand.java | 38 + .../googletv/GoogleTVConfiguration.java | 35 + .../googletv/GoogleTVConnectionManager.java | 1385 +++++++++++++++++ .../protocol/googletv/GoogleTVConstants.java | 44 + .../googletv/GoogleTVMessageParser.java | 336 ++++ .../protocol/googletv/GoogleTVRequest.java | 148 ++ .../protocol/googletv/GoogleTVUtils.java | 129 ++ .../protocol/shieldtv/ShieldTVCommand.java | 38 + .../shieldtv/ShieldTVConfiguration.java | 34 + .../shieldtv/ShieldTVConnectionManager.java | 1219 +++++++++++++++ .../protocol/shieldtv/ShieldTVConstants.java | 61 + .../shieldtv/ShieldTVMessageParser.java | 445 ++++++ .../protocol/shieldtv/ShieldTVRequest.java | 103 ++ .../internal/utils/AndroidTVPKI.java | 291 ++++ .../src/main/resources/OH-INF/addon/addon.xml | 11 + .../OH-INF/i18n/androidtv.properties | 69 + .../resources/OH-INF/thing/thing-types.xml | 192 +++ bundles/pom.xml | 1 + 30 files changed, 5749 insertions(+) create mode 100644 bundles/org.openhab.binding.androidtv/NOTICE create mode 100644 bundles/org.openhab.binding.androidtv/README.md create mode 100644 bundles/org.openhab.binding.androidtv/pom.xml create mode 100644 bundles/org.openhab.binding.androidtv/src/main/feature/feature.xml create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVBindingConstants.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVDynamicCommandDescriptionProvider.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVHandler.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVHandlerFactory.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/discovery/GoogleTVDiscoveryParticipant.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/discovery/ShieldTVDiscoveryParticipant.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVCommand.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConfiguration.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConnectionManager.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConstants.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVMessageParser.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVRequest.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVUtils.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVCommand.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConfiguration.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConnectionManager.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConstants.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVMessageParser.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVRequest.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/utils/AndroidTVPKI.java create mode 100644 bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/addon/addon.xml create mode 100644 bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/i18n/androidtv.properties create mode 100644 bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/thing/thing-types.xml diff --git a/CODEOWNERS b/CODEOWNERS index 899c818aa..ae42d31f2 100644 --- a/CODEOWNERS +++ b/CODEOWNERS @@ -24,6 +24,7 @@ /bundles/org.openhab.binding.ambientweather/ @mhilbush /bundles/org.openhab.binding.amplipi/ @kaikreuzer /bundles/org.openhab.binding.androiddebugbridge/ @GiviMAD +/bundles/org.openhab.binding.androidtv/ @morph166955 /bundles/org.openhab.binding.anel/ @paphko /bundles/org.openhab.binding.anthem/ @mhilbush /bundles/org.openhab.binding.astro/ @gerrieg diff --git a/bom/openhab-addons/pom.xml b/bom/openhab-addons/pom.xml index e19070fd2..54161aed4 100644 --- a/bom/openhab-addons/pom.xml +++ b/bom/openhab-addons/pom.xml @@ -111,6 +111,11 @@ org.openhab.binding.androiddebugbridge ${project.version} + + org.openhab.addons.bundles + org.openhab.binding.androidtv + ${project.version} + org.openhab.addons.bundles org.openhab.binding.anel diff --git a/bundles/org.openhab.binding.androidtv/NOTICE b/bundles/org.openhab.binding.androidtv/NOTICE new file mode 100644 index 000000000..38d625e34 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/NOTICE @@ -0,0 +1,13 @@ +This content is produced and maintained by the openHAB project. + +* Project home: https://www.openhab.org + +== Declared Project Licenses + +This program and the accompanying materials are made available under the terms +of the Eclipse Public License 2.0 which is available at +https://www.eclipse.org/legal/epl-2.0/. + +== Source Code + +https://github.com/openhab/openhab-addons diff --git a/bundles/org.openhab.binding.androidtv/README.md b/bundles/org.openhab.binding.androidtv/README.md new file mode 100644 index 000000000..b3132569f --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/README.md @@ -0,0 +1,484 @@ +# AndroidTV Binding + +This binding is designed to emulate different protocols to interact with the AndroidTV platform. +Currently it emulates both the Google Video App to interact with a variety of AndroidTVs for purposes of remote control. +It also currently emulates the Nvidia ShieldTV Android App to interact with an Nvidia ShieldTV for purposes of remote control. + +## Supported Things + +This binding supports two thing types: + +- **googletv** - An AndroidTV running Google Video +- **shieldtv** - An Nvidia ShieldTV + +## Discovery + +Both GoogleTVs and ShieldTVs should be added automatically to the inbox through the mDNS discovery process. + +In the case of the ShieldTV, openHAB will likely create an inbox entry for both a GoogleTV and a ShieldTV device. +Only the ShieldTV device should be configured, the GoogleTV can be ignored. +There is no benefit to configuring two things for a ShieldTV device. +This could cause undesired effects. + +## Binding Configuration + +This binding does not require any special configuration files. + +This binding does require a PIN login process (documented below) upon first connection. + +This binding requires GoogleTV to be installed on the device (https://play.google.com/store/apps/details?id=com.google.android.videos) + +## Thing Configuration + +There are three required fields to connect successfully to a ShieldTV. + +| Name | Type | Description | Default | Required | Advanced | +|------------------|---------|---------------------------------------|---------|----------|----------| +| ipAddress | text | IP address of the device | N/A | yes | no | +| keystore | text | Location of the Java Keystore | N/A | no | no | +| keystorePassword | text | Password of the Java Keystore | N/A | no | no | + +```java +Thing androidtv:shieldtv:livingroom [ ipAddress="192.168.1.2" ] +Thing androidtv:googletv:theater [ ipAddress="192.168.1.3" ] +``` + +## Channels + +| Channel | Type | Description | GoogleTV | ShieldTV | +|------------|--------|-----------------------------|----------|----------| +| keyboard | String | Keyboard Data Entry | RW | RW | +| keypress | String | Manual Key Press Entry | RW | RW | +| keycode | String | Direct KEYCODE Entry | RW | RW | +| pincode | String | PIN Code Entry | RW | RW | +| app | String | App Control | RO | RW | +| appname | String | App Name | N/A | RW | +| appurl | String | App URL | N/A | RW | +| player | Player | Player Control | RW | RW | +| power | Switch | Power Control | RW | RW | +| volume | Dimmer | Volume Control | RO | RO | +| mute | Switch | Mute Control | RW | RW | + + +```java +String ShieldTV_KEYBOARD "KEYBOARD [%s]" { channel = "androidtv:shieldtv:livingroom:keyboard" } +String ShieldTV_KEYPRESS "KEYPRESS [%s]" { channel = "androidtv:shieldtv:livingroom:keypress" } +String ShieldTV_KEYCODE "KEYCODE [%s]" { channel = "androidtv:shieldtv:livingroom:keycode" } +String ShieldTV_PINCODE "PINCODE [%s]" { channel = "androidtv:shieldtv:livingroom:pincode" } +String ShieldTV_APP "APP [%s]" { channel = "androidtv:shieldtv:livingroom:app" } +String ShieldTV_APPNAME "APPNAME [%s]" { channel = "androidtv:shieldtv:livingroom:appname" } +String ShieldTV_APPURL "APPURL [%s]" { channel = "androidtv:shieldtv:livingroom:appurl" } +Player ShieldTV_PLAYER "PLAYER [%s]" { channel = "androidtv:shieldtv:livingroom:player" } +Switch ShieldTV_POWER "POWER [%s]" { channel = "androidtv:shieldtv:livingroom:power" } +Dimmer ShieldTV_VOLUME "VOLUME [%s]" { channel = "androidtv:shieldtv:livingroom:volume" } +Switch ShieldTV_MUTE "MUTE [%s]" { channel = "androidtv:shieldtv:livingroom:mute" } + +String GoogleTV_KEYBOARD "KEYBOARD [%s]" { channel = "androidtv:googletv:theater:keyboard" } +String GoogleTV_KEYPRESS "KEYPRESS [%s]" { channel = "androidtv:googletv:theater:keypress" } +String GoogleTV_KEYCODE "KEYCODE [%s]" { channel = "androidtv:googletv:theater:keycode" } +String GoogleTV_PINCODE "PINCODE [%s]" { channel = "androidtv:googletv:theater:pincode" } +String GoogleTV_APP "APP [%s]" { channel = "androidtv:googletv:theater:app" } +Player GoogleTV_PLAYER "PLAYER [%s]" { channel = "androidtv:googletv:theater:player" } +Switch GoogleTV_POWER "POWER [%s]" { channel = "androidtv:googletv:theater:power" } +Dimmer GoogleTV_VOLUME "VOLUME [%s]" { channel = "androidtv:googletv:theater:volume" } +Switch GoogleTV_MUTE "MUTE [%s]" { channel = "androidtv:googletv:theater:mute" } +``` + +KEYPRESS will accept the following commands as strings (case sensitive): + +- KEY_UP +- KEY_DOWN +- KEY_RIGHT +- KEY_LEFT +- KEY_ENTER +- KEY_HOME +- KEY_BACK +- KEY_MENU +- KEY_PLAY +- KEY_PAUSE +- KEY_PLAYPAUSE +- KEY_STOP +- KEY_NEXT +- KEY_PREVIOUS +- KEY_REWIND +- KEY_FORWARD +- KEY_POWER +- KEY_GOOGLE +- KEY_VOLUP +- KEY_VOLDOWN +- KEY_MUTE +- KEY_SUBMIT + +The list above causes an instantanious "press and release" of each button. +If you would like to manually control the press and release of each you may append _PRESS and _RELEASE to the end of each. +(e.g. KEY_FORWARD_PRESS or KEY_FORWARD_RELEASE) + +You may also send an ASCII character as a single letter to simulate a key entry (e.g KEY_A, KEY_1, KEY_z). +Use KEY_SUBMIT when full text entry is complete to tell the shield to process the line. +KEY_SUBMIT is automatically sent by KEYBOARD when a command is sent to the channel. + +APP will display the currently active app as presented by the AndroidTV. +You may also send it a command of the app package name (e.g. com.google.android.youtube.tv) to start/change-to that app. + +KEYCODE values are listed at the bottom of this README. +NOTE: Not all KEYCODES work on all devices. Keycodes above 255 have not been tested. + +## Pin Code Process + +For the AndroidTV to be successfully accessed an on-screen PIN authentication is required on the first connection. + +To begin the PIN process, send the text "REQUEST" to the pincode channel while watching your AndroidTV. + +A 6 digit PIN should be displayed on the screen. + +To complete the PIN process, send the PIN displayed to the pincode channel. + +The display should return back to where it was originally. + +If you are on a ShieldTV you must run that process a second time to authenticate the GoogleTV protocol stack. + +This completes the PIN process. + +Upon reconnection (either from reconfiguration or a restart of OpenHAB), you should now see a message of "Login Successful" in openhab.log + +## Full Example + +```java +Thing androidtv:shieldtv:livingroom [ ipAddress="192.168.1.2" ] +Thing androidtv:googletv:theater [ ipAddress="192.168.1.3" ] +``` + +```java +String ShieldTV_KEYBOARD "KEYBOARD [%s]" { channel = "androidtv:shieldtv:livingroom:keyboard" } +String ShieldTV_KEYPRESS "KEYPRESS [%s]" { channel = "androidtv:shieldtv:livingroom:keypress" } +String ShieldTV_KEYCODE "KEYCODE [%s]" { channel = "androidtv:shieldtv:livingroom:keycode" } +String ShieldTV_PINCODE "PINCODE [%s]" { channel = "androidtv:shieldtv:livingroom:pincode" } +String ShieldTV_APP "APP [%s]" { channel = "androidtv:shieldtv:livingroom:app" } +String ShieldTV_APPNAME "APPNAME [%s]" { channel = "androidtv:shieldtv:livingroom:appname" } +String ShieldTV_APPURL "APPURL [%s]" { channel = "androidtv:shieldtv:livingroom:appurl" } +Player ShieldTV_PLAYER "PLAYER [%s]" { channel = "androidtv:shieldtv:livingroom:player" } +Switch ShieldTV_POWER "POWER [%s]" { channel = "androidtv:shieldtv:livingroom:power" } +Dimmer ShieldTV_VOLUME "VOLUME [%s]" { channel = "androidtv:shieldtv:livingroom:volume" } +Switch ShieldTV_MUTE "MUTE [%s]" { channel = "androidtv:shieldtv:livingroom:mute" } + +String GoogleTV_KEYBOARD "KEYBOARD [%s]" { channel = "androidtv:googletv:theater:keyboard" } +String GoogleTV_KEYPRESS "KEYPRESS [%s]" { channel = "androidtv:googletv:theater:keypress" } +String GoogleTV_KEYCODE "KEYCODE [%s]" { channel = "androidtv:googletv:theater:keycode" } +String GoogleTV_PINCODE "PINCODE [%s]" { channel = "androidtv:googletv:theater:pincode" } +String GoogleTV_APP "APP [%s]" { channel = "androidtv:googletv:theater:app" } +Player GoogleTV_PLAYER "PLAYER [%s]" { channel = "androidtv:googletv:theater:player" } +Switch GoogleTV_POWER "POWER [%s]" { channel = "androidtv:googletv:theater:power" } +Dimmer GoogleTV_VOLUME "VOLUME [%s]" { channel = "androidtv:googletv:theater:volume" } +Switch GoogleTV_MUTE "MUTE [%s]" { channel = "androidtv:googletv:theater:mute" } +``` + +## Google Keycodes + +| CODE | BUTTON | +|------|--------| +| 0 | KEYCODE_UNKNOWN | +| 1 | KEYCODE_SOFT_LEFT | +| 2 | KEYCODE_SOFT_RIGHT | +| 3 | KEYCODE_HOME | +| 4 | KEYCODE_BACK | +| 5 | KEYCODE_CALL | +| 6 | KEYCODE_ENDCALL | +| 7 | KEYCODE_0 | +| 8 | KEYCODE_1 | +| 9 | KEYCODE_2 | +| 10 | KEYCODE_3 | +| 11 | KEYCODE_4 | +| 12 | KEYCODE_5 | +| 13 | KEYCODE_6 | +| 14 | KEYCODE_7 | +| 15 | KEYCODE_8 | +| 16 | KEYCODE_9 | +| 17 | KEYCODE_STAR | +| 18 | KEYCODE_POUND | +| 19 | KEYCODE_DPAD_UP | +| 20 | KEYCODE_DPAD_DOWN | +| 21 | KEYCODE_DPAD_LEFT | +| 22 | KEYCODE_DPAD_RIGHT | +| 23 | KEYCODE_DPAD_CENTER | +| 24 | KEYCODE_VOLUME_UP | +| 25 | KEYCODE_VOLUME_DOWN | +| 26 | KEYCODE_POWER | +| 27 | KEYCODE_CAMERA | +| 28 | KEYCODE_CLEAR | +| 29 | KEYCODE_A | +| 30 | KEYCODE_B | +| 31 | KEYCODE_C | +| 32 | KEYCODE_D | +| 33 | KEYCODE_E | +| 34 | KEYCODE_F | +| 35 | KEYCODE_G | +| 36 | KEYCODE_H | +| 37 | KEYCODE_I | +| 38 | KEYCODE_J | +| 39 | KEYCODE_K | +| 40 | KEYCODE_L | +| 41 | KEYCODE_M | +| 42 | KEYCODE_N | +| 43 | KEYCODE_O | +| 44 | KEYCODE_P | +| 45 | KEYCODE_Q | +| 46 | KEYCODE_R | +| 47 | KEYCODE_S | +| 48 | KEYCODE_T | +| 49 | KEYCODE_U | +| 50 | KEYCODE_V | +| 51 | KEYCODE_W | +| 52 | KEYCODE_X | +| 53 | KEYCODE_Y | +| 54 | KEYCODE_Z | +| 55 | KEYCODE_COMMA | +| 56 | KEYCODE_PERIOD | +| 57 | KEYCODE_ALT_LEFT | +| 58 | KEYCODE_ALT_RIGHT | +| 59 | KEYCODE_SHIFT_LEFT | +| 60 | KEYCODE_SHIFT_RIGHT | +| 61 | KEYCODE_TAB | +| 62 | KEYCODE_SPACE | +| 63 | KEYCODE_SYM | +| 64 | KEYCODE_EXPLORER | +| 65 | KEYCODE_ENVELOPE | +| 66 | KEYCODE_ENTER | +| 67 | KEYCODE_DEL | +| 68 | KEYCODE_GRAVE | +| 69 | KEYCODE_MINUS | +| 70 | KEYCODE_EQUALS | +| 71 | KEYCODE_LEFT_BRACKET | +| 72 | KEYCODE_RIGHT_BRACKET | +| 73 | KEYCODE_BACKSLASH | +| 74 | KEYCODE_SEMICOLON | +| 75 | KEYCODE_APOSTROPHE | +| 76 | KEYCODE_SLASH | +| 77 | KEYCODE_AT | +| 78 | KEYCODE_NUM | +| 79 | KEYCODE_HEADSETHOOK | +| 80 | KEYCODE_FOCUS | +| 81 | KEYCODE_PLUS | +| 82 | KEYCODE_MENU | +| 83 | KEYCODE_NOTIFICATION | +| 84 | KEYCODE_SEARCH | +| 85 | KEYCODE_MEDIA_PLAY_PAUSE | +| 86 | KEYCODE_MEDIA_STOP | +| 87 | KEYCODE_MEDIA_NEXT | +| 88 | KEYCODE_MEDIA_PREVIOUS | +| 89 | KEYCODE_MEDIA_REWIND | +| 90 | KEYCODE_MEDIA_FAST_FORWARD | +| 91 | KEYCODE_MUTE | +| 92 | KEYCODE_PAGE_UP | +| 93 | KEYCODE_PAGE_DOWN | +| 94 | KEYCODE_PICTSYMBOLS | +| 95 | KEYCODE_SWITCH_CHARSET | +| 96 | KEYCODE_BUTTON_A | +| 97 | KEYCODE_BUTTON_B | +| 98 | KEYCODE_BUTTON_C | +| 99 | KEYCODE_BUTTON_X | +| 100 | KEYCODE_BUTTON_Y | +| 101 | KEYCODE_BUTTON_Z | +| 102 | KEYCODE_BUTTON_L1 | +| 103 | KEYCODE_BUTTON_R1 | +| 104 | KEYCODE_BUTTON_L2 | +| 105 | KEYCODE_BUTTON_R2 | +| 106 | KEYCODE_BUTTON_THUMBL | +| 107 | KEYCODE_BUTTON_THUMBR | +| 108 | KEYCODE_BUTTON_START | +| 109 | KEYCODE_BUTTON_SELECT | +| 110 | KEYCODE_BUTTON_MODE | +| 111 | KEYCODE_ESCAPE | +| 112 | KEYCODE_FORWARD_DEL | +| 113 | KEYCODE_CTRL_LEFT | +| 114 | KEYCODE_CTRL_RIGHT | +| 115 | KEYCODE_CAPS_LOCK | +| 116 | KEYCODE_SCROLL_LOCK | +| 117 | KEYCODE_META_LEFT | +| 118 | KEYCODE_META_RIGHT | +| 119 | KEYCODE_FUNCTION | +| 120 | KEYCODE_SYSRQ | +| 121 | KEYCODE_BREAK | +| 122 | KEYCODE_MOVE_HOME | +| 123 | KEYCODE_MOVE_END | +| 124 | KEYCODE_INSERT | +| 125 | KEYCODE_FORWARD | +| 126 | KEYCODE_MEDIA_PLAY | +| 127 | KEYCODE_MEDIA_PAUSE | +| 128 | KEYCODE_MEDIA_CLOSE | +| 129 | KEYCODE_MEDIA_EJECT | +| 130 | KEYCODE_MEDIA_RECORD | +| 131 | KEYCODE_F1 | +| 132 | KEYCODE_F2 | +| 133 | KEYCODE_F3 | +| 134 | KEYCODE_F4 | +| 135 | KEYCODE_F5 | +| 136 | KEYCODE_F6 | +| 137 | KEYCODE_F7 | +| 138 | KEYCODE_F8 | +| 139 | KEYCODE_F9 | +| 140 | KEYCODE_F10 | +| 141 | KEYCODE_F11 | +| 142 | KEYCODE_F12 | +| 143 | KEYCODE_NUM_LOCK | +| 144 | KEYCODE_NUMPAD_0 | +| 145 | KEYCODE_NUMPAD_1 | +| 146 | KEYCODE_NUMPAD_2 | +| 147 | KEYCODE_NUMPAD_3 | +| 148 | KEYCODE_NUMPAD_4 | +| 149 | KEYCODE_NUMPAD_5 | +| 150 | KEYCODE_NUMPAD_6 | +| 151 | KEYCODE_NUMPAD_7 | +| 152 | KEYCODE_NUMPAD_8 | +| 153 | KEYCODE_NUMPAD_9 | +| 154 | KEYCODE_NUMPAD_DIVIDE | +| 155 | KEYCODE_NUMPAD_MULTIPLY | +| 156 | KEYCODE_NUMPAD_SUBTRACT | +| 157 | KEYCODE_NUMPAD_ADD | +| 158 | KEYCODE_NUMPAD_DOT | +| 159 | KEYCODE_NUMPAD_COMMA | +| 160 | KEYCODE_NUMPAD_ENTER | +| 161 | KEYCODE_NUMPAD_EQUALS | +| 162 | KEYCODE_NUMPAD_LEFT_PAREN | +| 163 | KEYCODE_NUMPAD_RIGHT_PAREN | +| 164 | KEYCODE_VOLUME_MUTE | +| 165 | KEYCODE_INFO | +| 166 | KEYCODE_CHANNEL_UP | +| 167 | KEYCODE_CHANNEL_DOWN | +| 168 | KEYCODE_ZOOM_IN | +| 169 | KEYCODE_ZOOM_OUT | +| 170 | KEYCODE_TV | +| 171 | KEYCODE_WINDOW | +| 172 | KEYCODE_GUIDE | +| 173 | KEYCODE_DVR | +| 174 | KEYCODE_BOOKMARK | +| 175 | KEYCODE_CAPTIONS | +| 176 | KEYCODE_SETTINGS | +| 177 | KEYCODE_TV_POWER | +| 178 | KEYCODE_TV_INPUT | +| 179 | KEYCODE_STB_POWER | +| 180 | KEYCODE_STB_INPUT | +| 181 | KEYCODE_AVR_POWER | +| 182 | KEYCODE_AVR_INPUT | +| 183 | KEYCODE_PROG_RED | +| 184 | KEYCODE_PROG_GREEN | +| 185 | KEYCODE_PROG_YELLOW | +| 186 | KEYCODE_PROG_BLUE | +| 187 | KEYCODE_APP_SWITCH | +| 188 | KEYCODE_BUTTON_1 | +| 189 | KEYCODE_BUTTON_2 | +| 190 | KEYCODE_BUTTON_3 | +| 191 | KEYCODE_BUTTON_4 | +| 192 | KEYCODE_BUTTON_5 | +| 193 | KEYCODE_BUTTON_6 | +| 194 | KEYCODE_BUTTON_7 | +| 195 | KEYCODE_BUTTON_8 | +| 196 | KEYCODE_BUTTON_9 | +| 197 | KEYCODE_BUTTON_10 | +| 198 | KEYCODE_BUTTON_11 | +| 199 | KEYCODE_BUTTON_12 | +| 200 | KEYCODE_BUTTON_13 | +| 201 | KEYCODE_BUTTON_14 | +| 202 | KEYCODE_BUTTON_15 | +| 203 | KEYCODE_BUTTON_16 | +| 204 | KEYCODE_LANGUAGE_SWITCH | +| 205 | KEYCODE_MANNER_MODE | +| 206 | KEYCODE_3D_MODE | +| 207 | KEYCODE_CONTACTS | +| 208 | KEYCODE_CALENDAR | +| 209 | KEYCODE_MUSIC | +| 210 | KEYCODE_CALCULATOR | +| 211 | KEYCODE_ZENKAKU_HANKAKU | +| 212 | KEYCODE_EISU | +| 213 | KEYCODE_MUHENKAN | +| 214 | KEYCODE_HENKAN | +| 215 | KEYCODE_KATAKANA_HIRAGANA | +| 216 | KEYCODE_YEN | +| 217 | KEYCODE_RO | +| 218 | KEYCODE_KANA | +| 219 | KEYCODE_ASSIST | +| 220 | KEYCODE_BRIGHTNESS_DOWN | +| 221 | KEYCODE_BRIGHTNESS_UP | +| 222 | KEYCODE_MEDIA_AUDIO_TRACK | +| 223 | KEYCODE_SLEEP | +| 224 | KEYCODE_WAKEUP | +| 225 | KEYCODE_PAIRING | +| 226 | KEYCODE_MEDIA_TOP_MENU | +| 227 | KEYCODE_11 | +| 228 | KEYCODE_12 | +| 229 | KEYCODE_LAST_CHANNEL | +| 230 | KEYCODE_TV_DATA_SERVICE | +| 231 | KEYCODE_VOICE_ASSIST | +| 232 | KEYCODE_TV_RADIO_SERVICE | +| 233 | KEYCODE_TV_TELETEXT | +| 234 | KEYCODE_TV_NUMBER_ENTRY | +| 235 | KEYCODE_TV_TERRESTRIAL_ANALOG | +| 236 | KEYCODE_TV_TERRESTRIAL_DIGITAL | +| 237 | KEYCODE_TV_SATELLITE | +| 238 | KEYCODE_TV_SATELLITE_BS | +| 239 | KEYCODE_TV_SATELLITE_CS | +| 240 | KEYCODE_TV_SATELLITE_SERVICE | +| 241 | KEYCODE_TV_NETWORK | +| 242 | KEYCODE_TV_ANTENNA_CABLE | +| 243 | KEYCODE_TV_INPUT_HDMI_1 | +| 244 | KEYCODE_TV_INPUT_HDMI_2 | +| 245 | KEYCODE_TV_INPUT_HDMI_3 | +| 246 | KEYCODE_TV_INPUT_HDMI_4 | +| 247 | KEYCODE_TV_INPUT_COMPOSITE_1 | +| 248 | KEYCODE_TV_INPUT_COMPOSITE_2 | +| 249 | KEYCODE_TV_INPUT_COMPONENT_1 | +| 250 | KEYCODE_TV_INPUT_COMPONENT_2 | +| 251 | KEYCODE_TV_INPUT_VGA_1 | +| 252 | KEYCODE_TV_AUDIO_DESCRIPTION | +| 253 | KEYCODE_TV_AUDIO_DESCRIPTION_MIX_UP | +| 254 | KEYCODE_TV_AUDIO_DESCRIPTION_MIX_DOWN | +| 255 | KEYCODE_TV_ZOOM_MODE | +| 256 | KEYCODE_TV_CONTENTS_MENU | +| 257 | KEYCODE_TV_MEDIA_CONTEXT_MENU | +| 258 | KEYCODE_TV_TIMER_PROGRAMMING | +| 259 | KEYCODE_HELP | +| 260 | KEYCODE_NAVIGATE_PREVIOUS | +| 261 | KEYCODE_NAVIGATE_NEXT | +| 262 | KEYCODE_NAVIGATE_IN | +| 263 | KEYCODE_NAVIGATE_OUT | +| 264 | KEYCODE_STEM_PRIMARY | +| 265 | KEYCODE_STEM_1 | +| 266 | KEYCODE_STEM_2 | +| 267 | KEYCODE_STEM_3 | +| 268 | KEYCODE_DPAD_UP_LEFT | +| 269 | KEYCODE_DPAD_DOWN_LEFT | +| 270 | KEYCODE_DPAD_UP_RIGHT | +| 271 | KEYCODE_DPAD_DOWN_RIGHT | +| 272 | KEYCODE_MEDIA_SKIP_FORWARD | +| 273 | KEYCODE_MEDIA_SKIP_BACKWARD | +| 274 | KEYCODE_MEDIA_STEP_FORWARD | +| 275 | KEYCODE_MEDIA_STEP_BACKWARD | +| 276 | KEYCODE_SOFT_SLEEP | +| 277 | KEYCODE_CUT | +| 278 | KEYCODE_COPY | +| 279 | KEYCODE_PASTE | +| 280 | KEYCODE_SYSTEM_NAVIGATION_UP | +| 281 | KEYCODE_SYSTEM_NAVIGATION_DOWN | +| 282 | KEYCODE_SYSTEM_NAVIGATION_LEFT | +| 283 | KEYCODE_SYSTEM_NAVIGATION_RIGHT | +| 284 | KEYCODE_ALL_APPS | +| 285 | KEYCODE_REFRESH | +| 286 | KEYCODE_THUMBS_UP | +| 287 | KEYCODE_THUMBS_DOWN | +| 288 | KEYCODE_PROFILE_SWITCH | +| 289 | KEYCODE_VIDEO_APP_1 | +| 290 | KEYCODE_VIDEO_APP_2 | +| 291 | KEYCODE_VIDEO_APP_3 | +| 292 | KEYCODE_VIDEO_APP_4 | +| 293 | KEYCODE_VIDEO_APP_5 | +| 294 | KEYCODE_VIDEO_APP_6 | +| 295 | KEYCODE_VIDEO_APP_7 | +| 296 | KEYCODE_VIDEO_APP_8 | +| 297 | KEYCODE_FEATURED_APP_1 | +| 298 | KEYCODE_FEATURED_APP_2 | +| 299 | KEYCODE_FEATURED_APP_3 | +| 300 | KEYCODE_FEATURED_APP_4 | +| 301 | KEYCODE_DEMO_APP_1 | +| 302 | KEYCODE_DEMO_APP_2 | +| 303 | KEYCODE_DEMO_APP_3 | +| 304 | KEYCODE_DEMO_APP_4 | + diff --git a/bundles/org.openhab.binding.androidtv/pom.xml b/bundles/org.openhab.binding.androidtv/pom.xml new file mode 100644 index 000000000..e3a2936ac --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/pom.xml @@ -0,0 +1,32 @@ + + + + 4.0.0 + + + org.openhab.addons.bundles + org.openhab.addons.reactor.bundles + 4.0.0-SNAPSHOT + + + org.openhab.binding.androidtv + + openHAB Add-ons :: Bundles :: AndroidTV Binding + + + + org.bouncycastle + bcpkix-jdk15on + 1.52 + compile + + + org.bouncycastle + bcprov-jdk15on + 1.52 + compile + + + + diff --git a/bundles/org.openhab.binding.androidtv/src/main/feature/feature.xml b/bundles/org.openhab.binding.androidtv/src/main/feature/feature.xml new file mode 100644 index 000000000..ae915f51e --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/feature/feature.xml @@ -0,0 +1,10 @@ + + + mvn:org.openhab.core.features.karaf/org.openhab.core.features.karaf.openhab-core/${ohc.version}/xml/features + + + openhab-runtime-base + openhab-transport-mdns + mvn:org.openhab.addons.bundles/org.openhab.binding.androidtv/${project.version} + + diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVBindingConstants.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVBindingConstants.java new file mode 100644 index 000000000..0838f1753 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVBindingConstants.java @@ -0,0 +1,56 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal; + +import java.util.Set; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.openhab.core.thing.ThingTypeUID; + +/** + * The {@link AndroidTVBindingConstants} class defines common constants, which are + * used across the whole binding. + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class AndroidTVBindingConstants { + + private static final String BINDING_ID = "androidtv"; + + // List of all Thing Type UIDs + public static final ThingTypeUID THING_TYPE_GOOGLETV = new ThingTypeUID(BINDING_ID, "googletv"); + public static final ThingTypeUID THING_TYPE_SHIELDTV = new ThingTypeUID(BINDING_ID, "shieldtv"); + + public static final Set SUPPORTED_THING_TYPES = Set.of(THING_TYPE_GOOGLETV, THING_TYPE_SHIELDTV); + + // List of all Channel ids + public static final String CHANNEL_DEBUG = "debug"; + public static final String CHANNEL_KEYBOARD = "keyboard"; + public static final String CHANNEL_KEYPRESS = "keypress"; + public static final String CHANNEL_KEYCODE = "keycode"; + public static final String CHANNEL_PINCODE = "pincode"; + public static final String CHANNEL_APP = "app"; + public static final String CHANNEL_APPNAME = "appname"; + public static final String CHANNEL_APPURL = "appurl"; + public static final String CHANNEL_POWER = "power"; + public static final String CHANNEL_VOLUME = "volume"; + public static final String CHANNEL_MUTE = "mute"; + public static final String CHANNEL_PLAYER = "player"; + + // List of all config properties + public static final String PROPERTY_IP_ADDRESS = "ipAddress"; + + // List of all static String literals + public static final String PIN_REQUEST = "REQUEST"; +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVDynamicCommandDescriptionProvider.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVDynamicCommandDescriptionProvider.java new file mode 100644 index 000000000..9ac7a5fa1 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVDynamicCommandDescriptionProvider.java @@ -0,0 +1,43 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.openhab.core.events.EventPublisher; +import org.openhab.core.thing.binding.BaseDynamicCommandDescriptionProvider; +import org.openhab.core.thing.i18n.ChannelTypeI18nLocalizationService; +import org.openhab.core.thing.link.ItemChannelLinkRegistry; +import org.openhab.core.thing.type.DynamicCommandDescriptionProvider; +import org.osgi.service.component.annotations.Activate; +import org.osgi.service.component.annotations.Component; +import org.osgi.service.component.annotations.Reference; + +/** + * Dynamic provider of command options. + * + * @author Ben Rosenblum - Initial contribution + * + * Originally written for ADB by Christoph Weitkamp - Initial contribution + */ +@Component(service = { DynamicCommandDescriptionProvider.class, AndroidTVDynamicCommandDescriptionProvider.class }) +@NonNullByDefault +public class AndroidTVDynamicCommandDescriptionProvider extends BaseDynamicCommandDescriptionProvider { + @Activate + public AndroidTVDynamicCommandDescriptionProvider(final @Reference EventPublisher eventPublisher, // + final @Reference ItemChannelLinkRegistry itemChannelLinkRegistry, // + final @Reference ChannelTypeI18nLocalizationService channelTypeI18nLocalizationService) { + this.eventPublisher = eventPublisher; + this.itemChannelLinkRegistry = itemChannelLinkRegistry; + this.channelTypeI18nLocalizationService = channelTypeI18nLocalizationService; + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVHandler.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVHandler.java new file mode 100644 index 000000000..f43936324 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVHandler.java @@ -0,0 +1,261 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal; + +import static org.openhab.binding.androidtv.internal.AndroidTVBindingConstants.*; + +import java.util.ArrayList; +import java.util.List; +import java.util.Map; +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.ScheduledFuture; +import java.util.concurrent.TimeUnit; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.binding.androidtv.internal.protocol.googletv.GoogleTVConfiguration; +import org.openhab.binding.androidtv.internal.protocol.googletv.GoogleTVConnectionManager; +import org.openhab.binding.androidtv.internal.protocol.shieldtv.ShieldTVConfiguration; +import org.openhab.binding.androidtv.internal.protocol.shieldtv.ShieldTVConnectionManager; +import org.openhab.core.library.types.StringType; +import org.openhab.core.thing.ChannelUID; +import org.openhab.core.thing.Thing; +import org.openhab.core.thing.ThingStatus; +import org.openhab.core.thing.ThingStatusDetail; +import org.openhab.core.thing.ThingTypeUID; +import org.openhab.core.thing.binding.BaseThingHandler; +import org.openhab.core.types.Command; +import org.openhab.core.types.CommandOption; +import org.openhab.core.types.State; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * The {@link AndroidTVHandler} is responsible for handling commands, which are + * sent to one of the channels. + * + * Significant portions reused from Lutron binding with permission from Bob A. + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class AndroidTVHandler extends BaseThingHandler { + + private final Logger logger = LoggerFactory.getLogger(AndroidTVHandler.class); + + private @Nullable ShieldTVConnectionManager shieldtvConnectionManager; + private @Nullable GoogleTVConnectionManager googletvConnectionManager; + + private @Nullable ScheduledFuture monitorThingStatusJob; + private final Object monitorThingStatusJobLock = new Object(); + private static final int THING_STATUS_FREQUENCY = 250; + + private final AndroidTVDynamicCommandDescriptionProvider commandDescriptionProvider; + private final ThingTypeUID thingTypeUID; + private final String thingID; + + public AndroidTVHandler(Thing thing, AndroidTVDynamicCommandDescriptionProvider commandDescriptionProvider, + ThingTypeUID thingTypeUID) { + super(thing); + this.commandDescriptionProvider = commandDescriptionProvider; + this.thingTypeUID = thingTypeUID; + this.thingID = this.getThing().getUID().getId(); + } + + public void setThingProperty(String property, String value) { + thing.setProperty(property, value); + } + + public String getThingID() { + return this.thingID; + } + + public void updateChannelState(String channel, State state) { + updateState(channel, state); + } + + public ScheduledExecutorService getScheduler() { + return scheduler; + } + + public void updateCDP(String channelName, Map cdpMap) { + logger.trace("{} - Updating CDP for {}", this.thingID, channelName); + List commandOptions = new ArrayList(); + cdpMap.forEach((key, value) -> commandOptions.add(new CommandOption(key, value))); + logger.trace("{} - CDP List: {}", this.thingID, commandOptions); + commandDescriptionProvider.setCommandOptions(new ChannelUID(getThing().getUID(), channelName), commandOptions); + } + + private void monitorThingStatus() { + synchronized (monitorThingStatusJobLock) { + checkThingStatus(); + monitorThingStatusJob = scheduler.schedule(this::monitorThingStatus, THING_STATUS_FREQUENCY, + TimeUnit.MILLISECONDS); + } + } + + public void checkThingStatus() { + String statusMessage = ""; + boolean failed = false; + + GoogleTVConnectionManager googletvConnectionManager = this.googletvConnectionManager; + ShieldTVConnectionManager shieldtvConnectionManager = this.shieldtvConnectionManager; + + if (googletvConnectionManager != null) { + if (!googletvConnectionManager.getLoggedIn()) { + statusMessage = "GoogleTV: " + googletvConnectionManager.getStatusMessage(); + failed = true; + } else { + statusMessage = "GoogleTV: ONLINE"; + } + } + + if (THING_TYPE_SHIELDTV.equals(thingTypeUID)) { + if (shieldtvConnectionManager != null) { + if (!shieldtvConnectionManager.getLoggedIn()) { + statusMessage = statusMessage + " | ShieldTV: " + shieldtvConnectionManager.getStatusMessage(); + failed = true; + } else { + statusMessage = statusMessage + " | ShieldTV: ONLINE"; + } + } + } + + if (failed) { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.NONE, statusMessage); + } else { + updateStatus(ThingStatus.ONLINE); + } + } + + @Override + public void initialize() { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.NONE, "@text/offline.protocols-starting"); + + GoogleTVConfiguration googletvConfig = getConfigAs(GoogleTVConfiguration.class); + String ipAddress = googletvConfig.ipAddress; + + if (ipAddress.isBlank()) { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, + "@text/offline.googletv-address-not-specified"); + return; + } + + googletvConnectionManager = new GoogleTVConnectionManager(this, googletvConfig); + + if (THING_TYPE_SHIELDTV.equals(thingTypeUID)) { + ShieldTVConfiguration shieldtvConfig = getConfigAs(ShieldTVConfiguration.class); + ipAddress = shieldtvConfig.ipAddress; + + if (ipAddress.isBlank()) { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, + "@text/offline.shieldtv-address-not-specified"); + return; + } + + shieldtvConnectionManager = new ShieldTVConnectionManager(this, shieldtvConfig); + } + + monitorThingStatusJob = scheduler.schedule(this::monitorThingStatus, THING_STATUS_FREQUENCY, + TimeUnit.MILLISECONDS); + } + + @Override + public void handleCommand(ChannelUID channelUID, Command command) { + logger.trace("{} - Command received at handler: {} {}", this.thingID, channelUID.getId(), command); + + if (command.toString().equals("REFRESH")) { + // REFRESH causes issues on some channels. Block for now until implemented. + return; + } + + GoogleTVConnectionManager googletvConnectionManager = this.googletvConnectionManager; + ShieldTVConnectionManager shieldtvConnectionManager = this.shieldtvConnectionManager; + + if (CHANNEL_DEBUG.equals(channelUID.getId())) { + if (command instanceof StringType) { + if (command.toString().equals("GOOGLETV_HALT") && (googletvConnectionManager != null)) { + googletvConnectionManager.dispose(); + googletvConnectionManager = null; + } else if (command.toString().equals("GOOGLETV_START")) { + GoogleTVConfiguration googletvConfig = getConfigAs(GoogleTVConfiguration.class); + googletvConnectionManager = new GoogleTVConnectionManager(this, googletvConfig); + } else if (command.toString().equals("GOOGLETV_SHIM") && (googletvConnectionManager == null)) { + GoogleTVConfiguration googletvConfig = getConfigAs(GoogleTVConfiguration.class); + googletvConfig.shim = true; + googletvConnectionManager = new GoogleTVConnectionManager(this, googletvConfig); + } else if (command.toString().equals("SHIELDTV_HALT") && (shieldtvConnectionManager != null)) { + shieldtvConnectionManager.dispose(); + shieldtvConnectionManager = null; + } else if (command.toString().equals("SHIELDTV_START")) { + ShieldTVConfiguration shieldtvConfig = getConfigAs(ShieldTVConfiguration.class); + shieldtvConnectionManager = new ShieldTVConnectionManager(this, shieldtvConfig); + } else if (command.toString().equals("SHIELDTV_SHIM") && (shieldtvConnectionManager == null)) { + ShieldTVConfiguration shieldtvConfig = getConfigAs(ShieldTVConfiguration.class); + shieldtvConfig.shim = true; + shieldtvConnectionManager = new ShieldTVConnectionManager(this, shieldtvConfig); + } else if (command.toString().startsWith("GOOGLETV") && (googletvConnectionManager != null)) { + googletvConnectionManager.handleCommand(channelUID, command); + } else if (command.toString().startsWith("SHIELDTV") && (shieldtvConnectionManager != null)) { + shieldtvConnectionManager.handleCommand(channelUID, command); + } + } + return; + } + + if (THING_TYPE_SHIELDTV.equals(thingTypeUID) && (shieldtvConnectionManager != null)) { + if (CHANNEL_PINCODE.equals(channelUID.getId())) { + if (command instanceof StringType) { + if (!shieldtvConnectionManager.getLoggedIn()) { + shieldtvConnectionManager.handleCommand(channelUID, command); + return; + } + } + } else if (CHANNEL_APP.equals(channelUID.getId())) { + if (command instanceof StringType) { + shieldtvConnectionManager.handleCommand(channelUID, command); + return; + } + } + } + + if (googletvConnectionManager != null) { + googletvConnectionManager.handleCommand(channelUID, command); + return; + } + + logger.warn("{} - Commands All Failed. Please report this as a bug. {} {}", thingID, channelUID.getId(), + command); + } + + @Override + public void dispose() { + synchronized (monitorThingStatusJobLock) { + ScheduledFuture monitorThingStatusJob = this.monitorThingStatusJob; + if (monitorThingStatusJob != null) { + monitorThingStatusJob.cancel(true); + } + } + + GoogleTVConnectionManager googletvConnectionManager = this.googletvConnectionManager; + ShieldTVConnectionManager shieldtvConnectionManager = this.shieldtvConnectionManager; + + if (shieldtvConnectionManager != null) { + shieldtvConnectionManager.dispose(); + } + + if (googletvConnectionManager != null) { + googletvConnectionManager.dispose(); + } + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVHandlerFactory.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVHandlerFactory.java new file mode 100644 index 000000000..d5690ddc4 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/AndroidTVHandlerFactory.java @@ -0,0 +1,61 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal; + +import static org.openhab.binding.androidtv.internal.AndroidTVBindingConstants.*; + +import java.util.Set; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.core.thing.Thing; +import org.openhab.core.thing.ThingTypeUID; +import org.openhab.core.thing.binding.BaseThingHandlerFactory; +import org.openhab.core.thing.binding.ThingHandler; +import org.openhab.core.thing.binding.ThingHandlerFactory; +import org.osgi.service.component.annotations.Activate; +import org.osgi.service.component.annotations.Component; +import org.osgi.service.component.annotations.Reference; + +/** + * The {@link AndroidTVHandlerFactory} is responsible for creating things and thing + * handlers. + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +@Component(configurationPid = "binding.androidtv", service = ThingHandlerFactory.class) +public class AndroidTVHandlerFactory extends BaseThingHandlerFactory { + + private static final Set SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_GOOGLETV, + THING_TYPE_SHIELDTV); + + private final AndroidTVDynamicCommandDescriptionProvider commandDescriptionProvider; + + @Activate + public AndroidTVHandlerFactory( + final @Reference AndroidTVDynamicCommandDescriptionProvider commandDescriptionProvider) { + this.commandDescriptionProvider = commandDescriptionProvider; + } + + @Override + public boolean supportsThingType(ThingTypeUID thingTypeUID) { + return SUPPORTED_THING_TYPES_UIDS.contains(thingTypeUID); + } + + @Override + protected @Nullable ThingHandler createHandler(Thing thing) { + ThingTypeUID thingTypeUID = thing.getThingTypeUID(); + return new AndroidTVHandler(thing, commandDescriptionProvider, thingTypeUID); + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/discovery/GoogleTVDiscoveryParticipant.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/discovery/GoogleTVDiscoveryParticipant.java new file mode 100644 index 000000000..96c3852ce --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/discovery/GoogleTVDiscoveryParticipant.java @@ -0,0 +1,101 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.discovery; + +import static org.openhab.binding.androidtv.internal.AndroidTVBindingConstants.*; + +import java.net.InetAddress; +import java.util.Map; +import java.util.Set; + +import javax.jmdns.ServiceInfo; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.core.config.discovery.DiscoveryResult; +import org.openhab.core.config.discovery.DiscoveryResultBuilder; +import org.openhab.core.config.discovery.mdns.MDNSDiscoveryParticipant; +import org.openhab.core.thing.ThingTypeUID; +import org.openhab.core.thing.ThingUID; +import org.osgi.service.component.annotations.Component; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * Implementation of {@link MDNSDiscoveryParticipant} that will discover GOOGLETV(s). + * + * @author Ben Rosenblum - initial contribution + */ +@NonNullByDefault +@Component(service = MDNSDiscoveryParticipant.class, immediate = true, configurationPid = "discovery.googletv") +public class GoogleTVDiscoveryParticipant implements MDNSDiscoveryParticipant { + + private final Logger logger = LoggerFactory.getLogger(GoogleTVDiscoveryParticipant.class); + private static final String GOOGLETV_MDNS_SERVICE_TYPE = "_androidtvremote2._tcp.local."; + + @Override + public Set getSupportedThingTypeUIDs() { + return SUPPORTED_THING_TYPES; + } + + @Override + public String getServiceType() { + return GOOGLETV_MDNS_SERVICE_TYPE; + } + + @Override + public @Nullable DiscoveryResult createResult(@Nullable ServiceInfo service) { + if ((service == null) || !service.hasData()) { + return null; + } + + InetAddress[] ipAddresses = service.getInet4Addresses(); + + if (ipAddresses.length > 0) { + String ipAddress = ipAddresses[0].getHostAddress(); + String macAddress = service.getPropertyString("bt"); + + if (logger.isDebugEnabled()) { + String nice = service.getNiceTextString(); + String qualifiedName = service.getQualifiedName(); + logger.debug("GoogleTV mDNS discovery notified of GoogleTV mDNS service: {}", nice); + logger.trace("GoogleTV mDNS service qualifiedName: {}", qualifiedName); + logger.trace("GoogleTV mDNS service ipAddresses: {} ({})", ipAddresses, ipAddresses.length); + logger.trace("GoogleTV mDNS service selected ipAddress: {}", ipAddress); + logger.trace("GoogleTV mDNS service property macAddress: {}", macAddress); + } + + final ThingUID uid = getThingUID(service); + if (uid != null) { + final String id = uid.getId(); + final String label = service.getName() + " (" + id + ")"; + final Map properties = Map.of(PROPERTY_IP_ADDRESS, ipAddress); + + return DiscoveryResultBuilder.create(uid).withProperties(properties).withLabel(label).build(); + } else { + return null; + } + } else { + return null; + } + } + + @Override + public @Nullable ThingUID getThingUID(@Nullable ServiceInfo service) { + if ((service == null) || !service.hasData() || (service.getPropertyString("bt") == null)) { + return null; + } + + return new ThingUID(THING_TYPE_GOOGLETV, service.getPropertyString("bt").replace(":", "")); + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/discovery/ShieldTVDiscoveryParticipant.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/discovery/ShieldTVDiscoveryParticipant.java new file mode 100644 index 000000000..31c292522 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/discovery/ShieldTVDiscoveryParticipant.java @@ -0,0 +1,103 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.discovery; + +import static org.openhab.binding.androidtv.internal.AndroidTVBindingConstants.*; + +import java.net.InetAddress; +import java.util.Map; +import java.util.Set; + +import javax.jmdns.ServiceInfo; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.core.config.discovery.DiscoveryResult; +import org.openhab.core.config.discovery.DiscoveryResultBuilder; +import org.openhab.core.config.discovery.mdns.MDNSDiscoveryParticipant; +import org.openhab.core.thing.ThingTypeUID; +import org.openhab.core.thing.ThingUID; +import org.osgi.service.component.annotations.Component; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * Implementation of {@link MDNSDiscoveryParticipant} that will discover SHIELDTV(s). + * + * @author Ben Rosenblum - initial contribution + */ +@NonNullByDefault +@Component(service = MDNSDiscoveryParticipant.class, immediate = true, configurationPid = "discovery.shieldtv") +public class ShieldTVDiscoveryParticipant implements MDNSDiscoveryParticipant { + + private final Logger logger = LoggerFactory.getLogger(ShieldTVDiscoveryParticipant.class); + private static final String SHIELDTV_MDNS_SERVICE_TYPE = "_nv_shield_remote._tcp.local."; + + @Override + public Set getSupportedThingTypeUIDs() { + return SUPPORTED_THING_TYPES; + } + + @Override + public String getServiceType() { + return SHIELDTV_MDNS_SERVICE_TYPE; + } + + @Override + public @Nullable DiscoveryResult createResult(@Nullable ServiceInfo service) { + if (service == null || !service.hasData()) { + return null; + } + + InetAddress[] ipAddresses = service.getInet4Addresses(); + + if (ipAddresses.length > 0) { + String ipAddress = ipAddresses[0].getHostAddress(); + String serverId = service.getPropertyString("SERVER"); + String serverCapability = service.getPropertyString("SERVER_CAPABILITY"); + + if (logger.isDebugEnabled()) { + String nice = service.getNiceTextString(); + String qualifiedName = service.getQualifiedName(); + logger.debug("ShieldTV mDNS discovery notified of ShieldTV mDNS service: {}", nice); + logger.trace("ShieldTV mDNS service qualifiedName: {}", qualifiedName); + logger.trace("ShieldTV mDNS service ipAddresses: {} ({})", ipAddresses, ipAddresses.length); + logger.trace("ShieldTV mDNS service selected ipAddress: {}", ipAddress); + logger.trace("ShieldTV mDNS service property SERVER: {}", serverId); + logger.trace("ShieldTV mDNS service property SERVER_CAPABILITY: {}", serverCapability); + } + + final ThingUID uid = getThingUID(service); + if (uid != null) { + final String id = uid.getId(); + final String label = service.getName() + " (" + id + ")"; + final Map properties = Map.of(PROPERTY_IP_ADDRESS, ipAddress); + + return DiscoveryResultBuilder.create(uid).withProperties(properties).withLabel(label).build(); + } else { + return null; + } + } else { + return null; + } + } + + @Override + public @Nullable ThingUID getThingUID(@Nullable ServiceInfo service) { + if (service == null || !service.hasData() || (service.getPropertyString("SERVER") == null)) { + return null; + } + + return new ThingUID(THING_TYPE_SHIELDTV, service.getPropertyString("SERVER").substring(8)); + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVCommand.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVCommand.java new file mode 100644 index 000000000..6efcabfde --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVCommand.java @@ -0,0 +1,38 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.googletv; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * GoogleTVCommand represents a GoogleTV protocol command + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class GoogleTVCommand { + private String command; + + public GoogleTVCommand(String command) { + this.command = command; + } + + @Override + public String toString() { + return command; + } + + public boolean isEmpty() { + return command.isEmpty(); + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConfiguration.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConfiguration.java new file mode 100644 index 000000000..91fb0d685 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConfiguration.java @@ -0,0 +1,35 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.googletv; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * The {@link GoogleTVConfiguration} class contains fields mapping thing configuration parameters. + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class GoogleTVConfiguration { + + public String ipAddress = ""; + public int port = 6466; + public int reconnect; + public int heartbeat; + public String keystoreFileName = ""; + public String keystorePassword = ""; + public int delay = 0; + public boolean shim; + public boolean shimNewKeys; + public String mode = ""; +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConnectionManager.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConnectionManager.java new file mode 100644 index 000000000..d60465459 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConnectionManager.java @@ -0,0 +1,1385 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.googletv; + +import static org.openhab.binding.androidtv.internal.AndroidTVBindingConstants.*; +import static org.openhab.binding.androidtv.internal.protocol.googletv.GoogleTVConstants.*; + +import java.io.BufferedReader; +import java.io.BufferedWriter; +import java.io.File; +import java.io.FileInputStream; +import java.io.IOException; +import java.io.InputStreamReader; +import java.io.InterruptedIOException; +import java.io.OutputStreamWriter; +import java.math.BigInteger; +import java.net.ConnectException; +import java.net.InetSocketAddress; +import java.net.NoRouteToHostException; +import java.net.Socket; +import java.net.SocketAddress; +import java.net.SocketTimeoutException; +import java.net.UnknownHostException; +import java.nio.charset.StandardCharsets; +import java.security.GeneralSecurityException; +import java.security.KeyStore; +import java.security.NoSuchAlgorithmException; +import java.security.UnrecoverableKeyException; +import java.security.cert.Certificate; +import java.security.cert.CertificateEncodingException; +import java.security.cert.CertificateException; +import java.security.cert.X509Certificate; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.Future; +import java.util.concurrent.LinkedBlockingQueue; +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.ScheduledFuture; +import java.util.concurrent.TimeUnit; + +import javax.net.ssl.KeyManagerFactory; +import javax.net.ssl.SSLContext; +import javax.net.ssl.SSLServerSocket; +import javax.net.ssl.SSLServerSocketFactory; +import javax.net.ssl.SSLSession; +import javax.net.ssl.SSLSocket; +import javax.net.ssl.SSLSocketFactory; +import javax.net.ssl.TrustManager; +import javax.net.ssl.X509TrustManager; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.binding.androidtv.internal.AndroidTVHandler; +import org.openhab.binding.androidtv.internal.utils.AndroidTVPKI; +import org.openhab.core.OpenHAB; +import org.openhab.core.library.types.NextPreviousType; +import org.openhab.core.library.types.OnOffType; +import org.openhab.core.library.types.PercentType; +import org.openhab.core.library.types.PlayPauseType; +import org.openhab.core.library.types.RewindFastforwardType; +import org.openhab.core.library.types.StringType; +import org.openhab.core.thing.ChannelUID; +import org.openhab.core.types.Command; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * The {@link GoogleTVConnectionManager} is responsible for handling connections via the googletv protocol + * + * Significant portions reused from Lutron binding with permission from Bob A. + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class GoogleTVConnectionManager { + private static final int DEFAULT_RECONNECT_SECONDS = 60; + private static final int DEFAULT_HEARTBEAT_SECONDS = 5; + private static final long KEEPALIVE_TIMEOUT_SECONDS = 30; + private static final String DEFAULT_KEYSTORE_PASSWORD = "secret"; + private static final String DEFAULT_MODE = "NORMAL"; + private static final String PIN_MODE = "PIN"; + private static final int DEFAULT_PORT = 6466; + private static final int PIN_DELAY = 1000; + + private final Logger logger = LoggerFactory.getLogger(GoogleTVConnectionManager.class); + + private ScheduledExecutorService scheduler; + + private final AndroidTVHandler handler; + private GoogleTVConfiguration config; + + private @NonNullByDefault({}) SSLSocketFactory sslSocketFactory; + private @Nullable SSLSocket sslSocket; + private @Nullable BufferedWriter writer; + private @Nullable BufferedReader reader; + + private @NonNullByDefault({}) SSLServerSocketFactory sslServerSocketFactory; + private @Nullable Socket shimServerSocket; + private @Nullable BufferedWriter shimWriter; + private @Nullable BufferedReader shimReader; + + private @Nullable GoogleTVConnectionManager connectionManager; + private @Nullable GoogleTVConnectionManager childConnectionManager; + private @NonNullByDefault({}) GoogleTVMessageParser messageParser; + + private final BlockingQueue sendQueue = new LinkedBlockingQueue<>(); + private final BlockingQueue shimQueue = new LinkedBlockingQueue<>(); + + private @Nullable Future asyncInitializeTask; + private @Nullable Future shimAsyncInitializeTask; + + private @Nullable Thread senderThread; + private @Nullable Thread readerThread; + private @Nullable Thread shimSenderThread; + private @Nullable Thread shimReaderThread; + + private @Nullable ScheduledFuture keepAliveJob; + private @Nullable ScheduledFuture keepAliveReconnectJob; + private @Nullable ScheduledFuture connectRetryJob; + private final Object keepAliveReconnectLock = new Object(); + private final Object connectionLock = new Object(); + + private @Nullable ScheduledFuture deviceHealthJob; + private boolean isOnline = true; + + private StringBuffer sbReader = new StringBuffer(); + private StringBuffer sbShimReader = new StringBuffer(); + private String thisMsg = ""; + + private X509Certificate @Nullable [] shimX509ClientChain; + private Certificate @Nullable [] shimClientChain; + private Certificate @Nullable [] shimServerChain; + private Certificate @Nullable [] shimClientLocalChain; + + private boolean disposing = false; + private boolean isLoggedIn = false; + private String statusMessage = ""; + private String pinHash = ""; + private String shimPinHash = ""; + + private boolean power = false; + private String volCurr = "00"; + private String volMax = "ff"; + private boolean volMute = false; + private String audioMode = ""; + private String currentApp = ""; + private String manufacturer = ""; + private String model = ""; + private String androidVersion = ""; + private String remoteServer = ""; + private String remoteServerVersion = ""; + + private AndroidTVPKI androidtvPKI = new AndroidTVPKI(); + private byte[] encryptionKey; + + public GoogleTVConnectionManager(AndroidTVHandler handler, GoogleTVConfiguration config) { + messageParser = new GoogleTVMessageParser(this); + this.config = config; + this.handler = handler; + this.connectionManager = this; + this.scheduler = handler.getScheduler(); + this.encryptionKey = androidtvPKI.generateEncryptionKey(); + initialize(); + } + + public GoogleTVConnectionManager(AndroidTVHandler handler, GoogleTVConfiguration config, + GoogleTVConnectionManager connectionManager) { + messageParser = new GoogleTVMessageParser(this); + this.config = config; + this.handler = handler; + this.connectionManager = connectionManager; + this.scheduler = handler.getScheduler(); + this.encryptionKey = androidtvPKI.generateEncryptionKey(); + initialize(); + } + + public String getThingID() { + return handler.getThingID(); + } + + public void setManufacturer(String manufacturer) { + this.manufacturer = manufacturer; + handler.setThingProperty("manufacturer", manufacturer); + } + + public String getManufacturer() { + return manufacturer; + } + + public void setModel(String model) { + this.model = model; + handler.setThingProperty("model", model); + } + + public String getModel() { + return model; + } + + public void setAndroidVersion(String androidVersion) { + this.androidVersion = androidVersion; + handler.setThingProperty("androidVersion", androidVersion); + } + + public String getAndroidVersion() { + return androidVersion; + } + + public void setRemoteServer(String remoteServer) { + this.remoteServer = remoteServer; + handler.setThingProperty("remoteServer", remoteServer); + } + + public String getRemoteServer() { + return remoteServer; + } + + public void setRemoteServerVersion(String remoteServerVersion) { + this.remoteServerVersion = remoteServerVersion; + handler.setThingProperty("remoteServerVersion", remoteServerVersion); + } + + public String getRemoteServerVersion() { + return remoteServerVersion; + } + + public void setPower(boolean power) { + this.power = power; + logger.debug("{} - Setting power to {}", handler.getThingID(), power); + if (power) { + handler.updateChannelState(CHANNEL_POWER, OnOffType.ON); + } else { + handler.updateChannelState(CHANNEL_POWER, OnOffType.OFF); + } + } + + public boolean getPower() { + return power; + } + + public void setVolCurr(String volCurr) { + this.volCurr = volCurr; + int max = Integer.parseInt(this.volMax, 16); + int volume = ((Integer.parseInt(volCurr, 16) * 100) / max); + handler.updateChannelState(CHANNEL_VOLUME, new PercentType(volume)); + } + + public String getVolCurr() { + return volCurr; + } + + public void setVolMax(String volMax) { + this.volMax = volMax; + } + + public String getVolMax() { + return volMax; + } + + public void setVolMute(String volMute) { + if (DELIMITER_00.equals(volMute)) { + this.volMute = false; + handler.updateChannelState(CHANNEL_MUTE, OnOffType.OFF); + } else if (DELIMITER_01.equals(volMute)) { + this.volMute = true; + handler.updateChannelState(CHANNEL_MUTE, OnOffType.ON); + } + } + + public boolean getVolMute() { + return volMute; + } + + public void setAudioMode(String audioMode) { + this.audioMode = audioMode; + } + + public String getAudioMode() { + return audioMode; + } + + public void setCurrentApp(String currentApp) { + this.currentApp = currentApp; + handler.updateChannelState(CHANNEL_APP, new StringType(currentApp)); + } + + public String getStatusMessage() { + return statusMessage; + } + + private void setStatus(boolean isLoggedIn) { + if (isLoggedIn) { + setStatus(isLoggedIn, "ONLINE"); + } else { + setStatus(isLoggedIn, "UNKNOWN"); + } + } + + private void setStatus(boolean isLoggedIn, String statusMessage) { + if ((this.isLoggedIn != isLoggedIn) || (!this.statusMessage.equals(statusMessage))) { + this.isLoggedIn = isLoggedIn; + this.statusMessage = statusMessage; + handler.checkThingStatus(); + } + } + + public String getCurrentApp() { + return currentApp; + } + + public void setLoggedIn(boolean isLoggedIn) { + if (this.isLoggedIn != isLoggedIn) { + setStatus(isLoggedIn); + } + } + + public boolean getLoggedIn() { + return isLoggedIn; + } + + private boolean servicePing() { + int timeout = 500; + + SocketAddress socketAddress = new InetSocketAddress(config.ipAddress, config.port); + try (Socket socket = new Socket()) { + socket.connect(socketAddress, timeout); + return true; + } catch (ConnectException | SocketTimeoutException | NoRouteToHostException ignored) { + return false; + } catch (IOException ignored) { + // IOException is thrown by automatic close() of the socket. + // This should actually never return a value as we should return true above already + return true; + } + } + + private void checkHealth() { + boolean isOnline; + if (!isLoggedIn) { + isOnline = servicePing(); + } else { + isOnline = true; + } + logger.debug("{} - Device Health - Online: {} - Logged In: {} - Mode: {}", handler.getThingID(), isOnline, + isLoggedIn, config.mode); + if (isOnline != this.isOnline) { + this.isOnline = isOnline; + if (isOnline) { + logger.debug("{} - Device is back online. Attempting reconnection.", handler.getThingID()); + reconnect(); + } + } + } + + private void setShimX509ClientChain(X509Certificate @Nullable [] shimX509ClientChain) { + try { + this.shimX509ClientChain = shimX509ClientChain; + logger.trace("Setting shimX509ClientChain {}", config.port); + if (shimX509ClientChain != null && logger.isTraceEnabled()) { + for (int cert = 0; cert < shimX509ClientChain.length; cert++) { + logger.trace("Subject DN: {}", shimX509ClientChain[cert].getSubjectX500Principal()); + logger.trace("Issuer DN: {}", shimX509ClientChain[cert].getIssuerX500Principal()); + logger.trace("Serial number: {}", shimX509ClientChain[cert].getSerialNumber()); + logger.trace("Cert: {}", GoogleTVRequest + .decodeMessage(GoogleTVUtils.byteArrayToString(shimX509ClientChain[cert].getEncoded()))); + } + } + } catch (CertificateEncodingException e) { + logger.trace("setShimX509ClientChain CertificateEncodingException", e); + } + } + + private void startChildConnectionManager(int port, String mode) { + GoogleTVConfiguration childConfig = new GoogleTVConfiguration(); + childConfig.ipAddress = config.ipAddress; + childConfig.port = port; + childConfig.reconnect = config.reconnect; + childConfig.heartbeat = config.heartbeat; + childConfig.keystoreFileName = config.keystoreFileName; + childConfig.keystorePassword = config.keystorePassword; + childConfig.delay = config.delay; + childConfig.shim = config.shim; + childConfig.mode = mode; + logger.debug("{} - startChildConnectionManager parent config: {} {} {}", handler.getThingID(), config.port, + config.mode, config.shim); + logger.debug("{} - startChildConnectionManager child config: {} {} {}", handler.getThingID(), childConfig.port, + childConfig.mode, childConfig.shim); + childConnectionManager = new GoogleTVConnectionManager(this.handler, childConfig, this); + } + + private TrustManager[] defineNoOpTrustManager() { + return new TrustManager[] { new X509TrustManager() { + @Override + public void checkClientTrusted(final X509Certificate @Nullable [] chain, final @Nullable String authType) { + logger.debug("Assuming client certificate is valid"); + if (chain != null && logger.isTraceEnabled()) { + for (int cert = 0; cert < chain.length; cert++) { + logger.trace("Subject DN: {}", chain[cert].getSubjectX500Principal()); + logger.trace("Issuer DN: {}", chain[cert].getIssuerX500Principal()); + logger.trace("Serial number: {}", chain[cert].getSerialNumber()); + } + } + } + + @Override + public void checkServerTrusted(final X509Certificate @Nullable [] chain, final @Nullable String authType) { + logger.debug("Assuming server certificate is valid"); + if (chain != null && logger.isTraceEnabled()) { + for (int cert = 0; cert < chain.length; cert++) { + logger.trace("Subject DN: {}", chain[cert].getSubjectX500Principal()); + logger.trace("Issuer DN: {}", chain[cert].getIssuerX500Principal()); + logger.trace("Serial number: {}", chain[cert].getSerialNumber()); + } + } + } + + @Override + public X509Certificate @Nullable [] getAcceptedIssuers() { + X509Certificate[] x509ClientChain = shimX509ClientChain; + if (x509ClientChain != null && logger.isTraceEnabled()) { + logger.debug("Returning shimX509ClientChain for getAcceptedIssuers"); + for (int cert = 0; cert < x509ClientChain.length; cert++) { + logger.trace("Subject DN: {}", x509ClientChain[cert].getSubjectX500Principal()); + logger.trace("Issuer DN: {}", x509ClientChain[cert].getIssuerX500Principal()); + logger.trace("Serial number: {}", x509ClientChain[cert].getSerialNumber()); + } + return x509ClientChain; + } else { + logger.debug("Returning empty certificate for getAcceptedIssuers"); + return new X509Certificate[0]; + } + } + } }; + } + + private void initialize() { + SSLContext sslContext; + + String folderName = OpenHAB.getUserDataFolder() + "/androidtv"; + File folder = new File(folderName); + + if (!folder.exists()) { + logger.debug("Creating directory {}", folderName); + folder.mkdirs(); + } + + config.port = (config.port > 0) ? config.port : DEFAULT_PORT; + config.reconnect = (config.reconnect > 0) ? config.reconnect : DEFAULT_RECONNECT_SECONDS; + config.heartbeat = (config.heartbeat > 0) ? config.heartbeat : DEFAULT_HEARTBEAT_SECONDS; + config.delay = (config.delay < 0) ? 0 : config.delay; + config.shim = (config.shim) ? true : false; + config.shimNewKeys = (config.shimNewKeys) ? true : false; + config.mode = (!config.mode.equals("")) ? config.mode : DEFAULT_MODE; + + config.keystoreFileName = (!config.keystoreFileName.equals("")) ? config.keystoreFileName + : folderName + "/googletv." + ((config.shim) ? "shim." : "") + handler.getThing().getUID().getId() + + ".keystore"; + config.keystorePassword = (!config.keystorePassword.equals("")) ? config.keystorePassword + : DEFAULT_KEYSTORE_PASSWORD; + + androidtvPKI.setKeystoreFileName(config.keystoreFileName); + androidtvPKI.setAlias("nvidia"); + + if (config.mode.equals(DEFAULT_MODE)) { + deviceHealthJob = scheduler.scheduleWithFixedDelay(this::checkHealth, config.heartbeat, config.heartbeat, + TimeUnit.SECONDS); + } + + try { + File keystoreFile = new File(config.keystoreFileName); + + if (!keystoreFile.exists() || config.shimNewKeys) { + androidtvPKI.generateNewKeyPair(encryptionKey); + androidtvPKI.saveKeyStore(config.keystorePassword, this.encryptionKey); + } else { + androidtvPKI.loadFromKeyStore(config.keystorePassword, this.encryptionKey); + } + + logger.trace("{} - Initializing SSL Context", handler.getThingID()); + KeyManagerFactory kmf = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm()); + kmf.init(androidtvPKI.getKeyStore(config.keystorePassword, this.encryptionKey), + config.keystorePassword.toCharArray()); + + TrustManager[] trustManagers = defineNoOpTrustManager(); + + sslContext = SSLContext.getInstance("TLS"); + sslContext.init(kmf.getKeyManagers(), trustManagers, null); + + sslSocketFactory = sslContext.getSocketFactory(); + if (!config.shim) { + asyncInitializeTask = scheduler.submit(this::connect); + } else { + shimAsyncInitializeTask = scheduler.submit(this::shimInitialize); + } + } catch (NoSuchAlgorithmException | IOException e) { + setStatus(false, "Error initializing keystore"); + logger.debug("Error initializing keystore", e); + } catch (UnrecoverableKeyException e) { + setStatus(false, "Key unrecoverable with supplied password"); + } catch (GeneralSecurityException e) { + logger.debug("General security exception", e); + } catch (Exception e) { + logger.debug("General exception", e); + } + } + + public void connect() { + synchronized (connectionLock) { + if (isOnline || config.mode.equals(PIN_MODE)) { + try { + logger.debug("{} - Opening GoogleTV SSL connection to {}:{} {}", handler.getThingID(), + config.ipAddress, config.port, config.mode); + SSLSocket sslSocket = (SSLSocket) sslSocketFactory.createSocket(config.ipAddress, config.port); + sslSocket.startHandshake(); + this.shimServerChain = ((SSLSocket) sslSocket).getSession().getPeerCertificates(); + writer = new BufferedWriter( + new OutputStreamWriter(sslSocket.getOutputStream(), StandardCharsets.ISO_8859_1)); + reader = new BufferedReader( + new InputStreamReader(sslSocket.getInputStream(), StandardCharsets.ISO_8859_1)); + this.sslSocket = sslSocket; + this.sendQueue.clear(); + logger.debug("{} - Connection to {}:{} {} successful", handler.getThingID(), config.ipAddress, + config.port, config.mode); + } catch (UnknownHostException e) { + setStatus(false, "Unknown host"); + logger.debug("{} - Unknown host {}", handler.getThingID(), config.ipAddress); + return; + } catch (IllegalArgumentException e) { + // port out of valid range + setStatus(false, "Invalid port number"); + logger.debug("{} - Invalid port number {}:{}", handler.getThingID(), config.ipAddress, config.port); + return; + } catch (InterruptedIOException e) { + logger.debug("{} - Interrupted while establishing GoogleTV connection", handler.getThingID()); + Thread.currentThread().interrupt(); + return; + } catch (IOException e) { + String message = e.getMessage(); + if ((message != null) && (message.contains("certificate_unknown")) + && (!config.mode.equals(PIN_MODE)) && (!config.shim)) { + setStatus(false, "PIN Process Incomplete"); + logger.debug("{} - GoogleTV PIN Process Incomplete", handler.getThingID()); + reconnectTaskCancel(true); + startChildConnectionManager(this.config.port + 1, PIN_MODE); + } else if ((message != null) && (message.contains("certificate_unknown")) && (config.shim)) { + logger.debug("Shim cert_unknown I/O error while connecting: {}", e.getMessage()); + Socket shimServerSocket = this.shimServerSocket; + if (shimServerSocket != null) { + try { + shimServerSocket.close(); + } catch (IOException ex) { + logger.debug("Error closing GoogleTV SSL socket: {}", ex.getMessage()); + } + this.shimServerSocket = null; + } + } else { + setStatus(false, "Error opening GoogleTV SSL connection. Check log."); + logger.info("{} - Error opening GoogleTV SSL connection to {}:{} {}", handler.getThingID(), + config.ipAddress, config.port, e.getMessage()); + disconnect(false); + scheduleConnectRetry(config.reconnect); // Possibly a temporary problem. Try again later. + } + return; + } + + setStatus(false, "Initializing"); + + logger.trace("{} - Starting Reader Thread for {}:{}", handler.getThingID(), config.ipAddress, + config.port); + + Thread readerThread = new Thread(this::readerThreadJob, "GoogleTV reader " + handler.getThingID()); + readerThread.setDaemon(true); + readerThread.start(); + this.readerThread = readerThread; + + logger.trace("{} - Starting Sender Thread for {}:{}", handler.getThingID(), config.ipAddress, + config.port); + + Thread senderThread = new Thread(this::senderThreadJob, "GoogleTV sender " + handler.getThingID()); + senderThread.setDaemon(true); + senderThread.start(); + this.senderThread = senderThread; + + logger.trace("{} - Checking for PIN MODE for {}:{} {}", handler.getThingID(), config.ipAddress, + config.port, config.mode); + + if (config.mode.equals(PIN_MODE)) { + logger.trace("{} - Sending PIN Login to {}:{}", handler.getThingID(), config.ipAddress, + config.port); + // Send app name and device name + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage(GoogleTVRequest.loginRequest(1)))); + // Unknown but required + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage(GoogleTVRequest.loginRequest(2)))); + // Don't send pin request yet, let user send REQUEST via PINCODE channel + } else { + logger.trace("{} - Not PIN Mode {}:{} {}", handler.getThingID(), config.ipAddress, config.port, + config.mode); + } + } else { + scheduleConnectRetry(config.reconnect); // Possibly a temporary problem. Try again later. + } + } + } + + public void shimInitialize() { + synchronized (connectionLock) { + AndroidTVPKI shimPKI = new AndroidTVPKI(); + byte[] shimEncryptionKey = shimPKI.generateEncryptionKey(); + SSLContext sslContext; + + try { + shimPKI.generateNewKeyPair(shimEncryptionKey); + // Move this to PKI. Shim requires a trusted cert chain in the keystore. + KeyStore keystore = KeyStore.getInstance("JKS"); + FileInputStream keystoreInputStream = new FileInputStream(config.keystoreFileName); + keystore.load(keystoreInputStream, config.keystorePassword.toCharArray()); + + KeyManagerFactory kmf = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm()); + kmf.init(keystore, config.keystorePassword.toCharArray()); + TrustManager[] trustManagers = defineNoOpTrustManager(); + + sslContext = SSLContext.getInstance("TLS"); + sslContext.init(kmf.getKeyManagers(), trustManagers, null); + this.sslServerSocketFactory = sslContext.getServerSocketFactory(); + + logger.trace("Opening GoogleTV shim on port {}", config.port); + SSLServerSocket sslServerSocket = (SSLServerSocket) this.sslServerSocketFactory + .createServerSocket(config.port); + if (this.config.mode.equals(DEFAULT_MODE)) { + sslServerSocket.setNeedClientAuth(true); + } else { + sslServerSocket.setWantClientAuth(true); + } + + while (true) { + logger.trace("Waiting for shim connection... {}", config.port); + if (this.config.mode.equals(DEFAULT_MODE) && (childConnectionManager == null)) { + logger.trace("Starting childConnectionManager {}", config.port); + startChildConnectionManager(this.config.port + 1, PIN_MODE); + } + SSLSocket serverSocket = (SSLSocket) sslServerSocket.accept(); + logger.trace("shimInitialize accepted {}", config.port); + try { + serverSocket.startHandshake(); + logger.trace("shimInitialize startHandshake {}", config.port); + connect(); + logger.trace("shimInitialize connected {}", config.port); + + SSLSession session = serverSocket.getSession(); + Certificate[] cchain2 = session.getPeerCertificates(); + this.shimClientChain = cchain2; + Certificate[] cchain3 = session.getLocalCertificates(); + this.shimClientLocalChain = cchain3; + + X509Certificate[] shimX509ClientChain = new X509Certificate[cchain2.length]; + + for (int i = 0; i < cchain2.length; i++) { + logger.trace("Connection from: {}", + ((X509Certificate) cchain2[i]).getSubjectX500Principal()); + shimX509ClientChain[i] = ((X509Certificate) cchain2[i]); + } + + if (this.config.mode.equals(PIN_MODE)) { + this.shimX509ClientChain = shimX509ClientChain; + GoogleTVConnectionManager connectionManager = this.connectionManager; + if (connectionManager != null) { + connectionManager.setShimX509ClientChain(shimX509ClientChain); + } + } + + if (cchain3 != null) { + for (int i = 0; i < cchain3.length; i++) { + logger.trace("Connection from: {}", + ((X509Certificate) cchain3[i]).getSubjectX500Principal()); + } + } + + logger.trace("Peer host is {}", session.getPeerHost()); + logger.trace("Cipher is {}", session.getCipherSuite()); + logger.trace("Protocol is {}", session.getProtocol()); + logger.trace("ID is {}", new BigInteger(session.getId())); + logger.trace("Session created in {}", session.getCreationTime()); + logger.trace("Session accessed in {}", session.getLastAccessedTime()); + + shimWriter = new BufferedWriter( + new OutputStreamWriter(serverSocket.getOutputStream(), StandardCharsets.ISO_8859_1)); + shimReader = new BufferedReader( + new InputStreamReader(serverSocket.getInputStream(), StandardCharsets.ISO_8859_1)); + this.shimServerSocket = serverSocket; + this.shimQueue.clear(); + + Thread readerThread = new Thread(this::shimReaderThreadJob, "GoogleTV shim reader"); + readerThread.setDaemon(true); + readerThread.start(); + this.shimReaderThread = readerThread; + + Thread senderThread = new Thread(this::shimSenderThreadJob, "GoogleTV shim sender"); + senderThread.setDaemon(true); + senderThread.start(); + this.shimSenderThread = senderThread; + } catch (Exception e) { + logger.trace("Shim initalization exception {}", config.port); + logger.trace("Shim initalization exception", e); + } + } + } catch (Exception e) { + logger.trace("Shim initalization exception {}", config.port); + logger.trace("Shim initalization exception", e); + + return; + } + } + } + + private void scheduleConnectRetry(long waitSeconds) { + logger.trace("{} - Scheduling GoogleTV connection retry in {} seconds", handler.getThingID(), waitSeconds); + connectRetryJob = scheduler.schedule(this::connect, waitSeconds, TimeUnit.SECONDS); + } + + /** + * Disconnect from bridge, cancel retry and keepalive jobs, stop reader and writer threads, and + * clean up. + * + * @param interruptAll Set if reconnect task should be interrupted if running. Should be false when calling from + * connect or reconnect, and true when calling from dispose. + */ + private void disconnect(boolean interruptAll) { + synchronized (connectionLock) { + logger.debug("{} - Disconnecting GoogleTV", handler.getThingID()); + + this.isLoggedIn = false; + + ScheduledFuture connectRetryJob = this.connectRetryJob; + if (connectRetryJob != null) { + connectRetryJob.cancel(true); + } + ScheduledFuture keepAliveJob = this.keepAliveJob; + if (keepAliveJob != null) { + keepAliveJob.cancel(true); + } + reconnectTaskCancel(interruptAll); // May be called from keepAliveReconnectJob thread + + Thread senderThread = this.senderThread; + if (senderThread != null && senderThread.isAlive()) { + senderThread.interrupt(); + } + + Thread readerThread = this.readerThread; + if (readerThread != null && readerThread.isAlive()) { + readerThread.interrupt(); + } + + Thread shimSenderThread = this.shimSenderThread; + if (shimSenderThread != null && shimSenderThread.isAlive()) { + shimSenderThread.interrupt(); + } + + Thread shimReaderThread = this.shimReaderThread; + if (shimReaderThread != null && shimReaderThread.isAlive()) { + shimReaderThread.interrupt(); + } + + SSLSocket sslSocket = this.sslSocket; + if (sslSocket != null) { + try { + sslSocket.close(); + } catch (IOException e) { + logger.debug("Error closing GoogleTV SSL socket: {}", e.getMessage()); + } + this.sslSocket = null; + } + BufferedReader reader = this.reader; + if (reader != null) { + try { + reader.close(); + } catch (IOException e) { + logger.debug("Error closing reader: {}", e.getMessage()); + } + } + BufferedWriter writer = this.writer; + if (writer != null) { + try { + writer.close(); + } catch (IOException e) { + logger.debug("Error closing writer: {}", e.getMessage()); + } + } + + Socket shimServerSocket = this.shimServerSocket; + if (shimServerSocket != null) { + try { + shimServerSocket.close(); + } catch (IOException e) { + logger.debug("Error closing GoogleTV SSL socket: {}", e.getMessage()); + } + this.shimServerSocket = null; + } + BufferedReader shimReader = this.shimReader; + if (shimReader != null) { + try { + shimReader.close(); + } catch (IOException e) { + logger.debug("Error closing shimReader: {}", e.getMessage()); + } + } + BufferedWriter shimWriter = this.shimWriter; + if (shimWriter != null) { + try { + shimWriter.close(); + } catch (IOException e) { + logger.debug("Error closing shimWriter: {}", e.getMessage()); + } + } + } + } + + private void reconnect() { + synchronized (connectionLock) { + if (!this.disposing) { + logger.debug("{} - Attempting to reconnect to the GoogleTV", handler.getThingID()); + setStatus(false, "reconnecting"); + disconnect(false); + connect(); + } + } + } + + /** + * Method executed by the message sender thread (senderThread) + */ + private void senderThreadJob() { + logger.debug("{} - Command sender thread started {}", handler.getThingID(), config.port); + try { + while (!Thread.currentThread().isInterrupted() && writer != null) { + GoogleTVCommand command = sendQueue.take(); + + try { + BufferedWriter writer = this.writer; + if (writer != null) { + logger.trace("{} - Raw GoogleTV command decodes as: {}", handler.getThingID(), + GoogleTVRequest.decodeMessage(command.toString())); + writer.write(command.toString()); + writer.flush(); + } + } catch (InterruptedIOException e) { + logger.debug("Interrupted while sending to GoogleTV"); + setStatus(false, "Interrupted"); + break; // exit loop and terminate thread + } catch (IOException e) { + logger.warn("{} - Communication error, will try to reconnect GoogleTV. Error: {}", + handler.getThingID(), e.getMessage()); + setStatus(false, "Communication error, will try to reconnect"); + sendQueue.add(command); // Requeue command + this.isLoggedIn = false; + reconnect(); + break; // reconnect() will start a new thread; terminate this one + } + if (config.delay > 0) { + Thread.sleep(config.delay); // introduce delay to throttle send rate + } + } + } catch (InterruptedException e) { + Thread.currentThread().interrupt(); + } finally { + logger.debug("{} - Command sender thread exiting {}", handler.getThingID(), config.port); + } + } + + private void shimSenderThreadJob() { + logger.debug("Shim sender thread started"); + try { + while (!Thread.currentThread().isInterrupted() && shimWriter != null) { + GoogleTVCommand command = shimQueue.take(); + + try { + BufferedWriter writer = this.shimWriter; + if (writer != null) { + logger.trace("Shim received from google: {}", + GoogleTVRequest.decodeMessage(command.toString())); + writer.write(command.toString()); + writer.flush(); + } + } catch (InterruptedIOException e) { + logger.debug("Shim interrupted while sending."); + break; // exit loop and terminate thread + } catch (IOException e) { + logger.warn("Shim communication error. Error: {}", e.getMessage()); + break; // reconnect() will start a new thread; terminate this one + } + } + } catch (InterruptedException e) { + Thread.currentThread().interrupt(); + } finally { + logger.debug("Command sender thread exiting"); + } + } + + /** + * Method executed by the message reader thread (readerThread) + */ + private void readerThreadJob() { + logger.debug("{} - Message reader thread started {}", handler.getThingID(), config.port); + try { + BufferedReader reader = this.reader; + int length = 0; + int current = 0; + while (!Thread.interrupted() && reader != null) { + thisMsg = GoogleTVRequest.fixMessage(Integer.toHexString(reader.read())); + if (HARD_DROP.equals(thisMsg)) { + // Google has crashed the connection. Disconnect hard. + logger.debug("{} - readerThreadJob received ffffffff. Disconnecting hard.", handler.getThingID()); + this.isLoggedIn = false; + reconnect(); + break; + } + if (length == 0) { + length = Integer.parseInt(thisMsg.toString(), 16); + logger.trace("{} - readerThreadJob message length {}", handler.getThingID(), length); + current = 0; + sbReader = new StringBuffer(); + sbReader.append(thisMsg.toString()); + } else { + sbReader.append(thisMsg.toString()); + current += 1; + } + + if ((length > 0) && (current == length)) { + logger.trace("{} - GoogleTV Message: {} {}", handler.getThingID(), length, sbReader.toString()); + messageParser.handleMessage(sbReader.toString()); + if (config.shim) { + String thisCommand = interceptMessages(sbReader.toString()); + shimQueue.add(new GoogleTVCommand(GoogleTVRequest.encodeMessage(thisCommand))); + } + length = 0; + } + } + } catch (InterruptedIOException e) { + logger.debug("Interrupted while reading"); + setStatus(false, "Interrupted"); + } catch (IOException e) { + String message = e.getMessage(); + if ((message != null) && (message.contains("certificate_unknown")) && (!config.mode.equals(PIN_MODE)) + && (!config.shim)) { + setStatus(false, "PIN Process Incomplete"); + logger.debug("{} - GoogleTV PIN Process Incomplete", handler.getThingID()); + reconnectTaskCancel(true); + startChildConnectionManager(this.config.port + 1, PIN_MODE); + } else if ((message != null) && (message.contains("certificate_unknown")) && (config.shim)) { + logger.debug("Shim cert_unknown I/O error while reading from stream: {}", e.getMessage()); + Socket shimServerSocket = this.shimServerSocket; + if (shimServerSocket != null) { + try { + shimServerSocket.close(); + } catch (IOException ex) { + logger.debug("Error closing GoogleTV SSL socket: {}", ex.getMessage()); + } + this.shimServerSocket = null; + } + } else { + logger.debug("I/O error while reading from stream: {}", e.getMessage()); + setStatus(false, "I/O Error"); + } + } catch (RuntimeException e) { + logger.warn("Runtime exception in reader thread", e); + setStatus(false, "Runtime exception"); + } finally { + logger.debug("{} - Message reader thread exiting {}", handler.getThingID(), config.port); + } + } + + private String interceptMessages(String message) { + if (message.startsWith("080210c801c202", 2)) { + // intercept PIN hash and replace with valid shim hash + int length = this.pinHash.length() / 2; + String len1 = GoogleTVRequest.fixMessage(Integer.toHexString(length + 2)); + String len2 = GoogleTVRequest.fixMessage(Integer.toHexString(length)); + String reply = "080210c801c202" + len1 + "0a" + len2 + this.pinHash; + String replyLength = GoogleTVRequest.fixMessage(Integer.toHexString(reply.length() / 2)); + String finalReply = replyLength + reply; + logger.trace("Message Intercepted: {}", message); + logger.trace("Message chagnged to: {}", finalReply); + return finalReply; + } else if (message.startsWith("080210c801ca02", 2)) { + // intercept PIN hash and replace with valid shim hash + int length = this.shimPinHash.length() / 2; + String len1 = GoogleTVRequest.fixMessage(Integer.toHexString(length + 2)); + String len2 = GoogleTVRequest.fixMessage(Integer.toHexString(length)); + String reply = "080210c801ca02" + len1 + "0a" + len2 + this.shimPinHash; + String replyLength = GoogleTVRequest.fixMessage(Integer.toHexString(reply.length() / 2)); + String finalReply = replyLength + reply; + logger.trace("Message Intercepted: {}", message); + logger.trace("Message chagnged to: {}", finalReply); + return finalReply; + } else { + // don't intercept message + return message; + } + } + + private void shimReaderThreadJob() { + logger.debug("Shim reader thread started {}", config.port); + try { + BufferedReader reader = this.shimReader; + String thisShimMsg = ""; + int length = 0; + int current = 0; + while (!Thread.interrupted() && reader != null) { + thisShimMsg = GoogleTVRequest.fixMessage(Integer.toHexString(reader.read())); + if (HARD_DROP.equals(thisShimMsg)) { + // Google has crashed the connection. Disconnect hard. + disconnect(false); + break; + } + if (length == 0) { + length = Integer.parseInt(thisShimMsg.toString(), 16); + logger.trace("shimReaderThreadJob message length {}", length); + current = 0; + sbShimReader = new StringBuffer(); + sbShimReader.append(thisShimMsg.toString()); + } else { + sbShimReader.append(thisShimMsg.toString()); + current += 1; + } + if ((length > 0) && (current == length)) { + logger.trace("Shim GoogleTV Message: {} {}", length, sbShimReader.toString()); + String thisCommand = interceptMessages(sbShimReader.toString()); + sendQueue.add(new GoogleTVCommand(GoogleTVRequest.encodeMessage(thisCommand))); + length = 0; + } + } + } catch (InterruptedIOException e) { + logger.debug("Interrupted while reading"); + setStatus(false, "Interrupted"); + } catch (IOException e) { + logger.debug("I/O error while reading from stream: {}", e.getMessage()); + setStatus(false, "I/O Error"); + } catch (RuntimeException e) { + logger.warn("Runtime exception in reader thread", e); + setStatus(false, "Runtime exception"); + } finally { + logger.debug("Shim message reader thread exiting {}", config.port); + } + } + + public void sendKeepAlive(String request) { + String keepalive = GoogleTVRequest.encodeMessage(GoogleTVRequest.keepAlive(request)); + logger.debug("{} - Sending GoogleTV keepalive - request {} - response {}", handler.getThingID(), request, + GoogleTVRequest.decodeMessage(keepalive)); + sendCommand(new GoogleTVCommand(keepalive)); + reconnectTaskSchedule(); + } + + /** + * Schedules the reconnect task keepAliveReconnectJob to execute in KEEPALIVE_TIMEOUT_SECONDS. This should + * be + * cancelled by calling reconnectTaskCancel() if a valid response is received from the bridge. + */ + private void reconnectTaskSchedule() { + synchronized (keepAliveReconnectLock) { + logger.trace("{} - Scheduling Reconnect Job for {}", handler.getThingID(), KEEPALIVE_TIMEOUT_SECONDS); + keepAliveReconnectJob = scheduler.schedule(this::keepAliveTimeoutExpired, KEEPALIVE_TIMEOUT_SECONDS, + TimeUnit.SECONDS); + } + } + + /** + * Cancels the reconnect task keepAliveReconnectJob. + */ + private void reconnectTaskCancel(boolean interrupt) { + synchronized (keepAliveReconnectLock) { + ScheduledFuture keepAliveReconnectJob = this.keepAliveReconnectJob; + if (keepAliveReconnectJob != null) { + logger.trace("{} - Canceling GoogleTV scheduled reconnect job.", handler.getThingID()); + keepAliveReconnectJob.cancel(interrupt); + this.keepAliveReconnectJob = null; + } + } + } + + /** + * Executed by keepAliveReconnectJob if it is not cancelled by the LEAP message parser calling + * validMessageReceived() which in turn calls reconnectTaskCancel(). + */ + private void keepAliveTimeoutExpired() { + logger.debug("{} - GoogleTV keepalive response timeout expired. Initiating reconnect.", handler.getThingID()); + reconnect(); + } + + public void validMessageReceived() { + reconnectTaskCancel(true); // Got a good message, so cancel reconnect task. + } + + public void finishPinProcess() { + GoogleTVConnectionManager connectionManager = this.connectionManager; + GoogleTVConnectionManager childConnectionManager = this.childConnectionManager; + if ((connectionManager != null) && (config.mode.equals(PIN_MODE)) && (!config.shim)) { + disconnect(false); + connectionManager.finishPinProcess(); + } else if ((childConnectionManager != null) && (config.mode.equals(DEFAULT_MODE)) && (!config.shim)) { + childConnectionManager.dispose(); + reconnect(); + } + } + + public void sendCommand(GoogleTVCommand command) { + if ((!config.shim) && (!command.isEmpty())) { + int length = command.toString().length(); + String hexLength = GoogleTVRequest.encodeMessage(GoogleTVRequest.fixMessage(Integer.toHexString(length))); + String message = hexLength + command.toString(); + GoogleTVCommand lenCommand = new GoogleTVCommand(message); + sendQueue.add(lenCommand); + } + } + + public void sendShim(GoogleTVCommand command) { + if (!command.isEmpty()) { + shimQueue.add(command); + } + } + + public void handleCommand(ChannelUID channelUID, Command command) { + logger.debug("{} - Command received: {}", handler.getThingID(), channelUID.getId()); + + if (CHANNEL_KEYPRESS.equals(channelUID.getId())) { + if (command instanceof StringType) { + if (command.toString().length() == 5) { + // Account for KEY_(ASCII Character) + String keyPress = "aa01071a0512031a01" + + GoogleTVRequest.decodeMessage(new String("" + command.toString().charAt(4))); + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage(keyPress))); + return; + } + + String message = ""; + String suffix = ""; + String shortCommand = command.toString(); + if (command.toString().endsWith("_PRESS")) { + suffix = "1001"; + shortCommand = "KEY_" + command.toString().split("_")[1]; + } else if (command.toString().endsWith("_RELEASE")) { + suffix = "1002"; + shortCommand = "KEY_" + command.toString().split("_")[1]; + } else { + suffix = "1003"; + } + + switch (shortCommand) { + case "KEY_UP": + message = "52040813" + suffix; + break; + case "KEY_DOWN": + message = "52040814" + suffix; + break; + case "KEY_RIGHT": + message = "52040816" + suffix; + break; + case "KEY_LEFT": + message = "52040815" + suffix; + break; + case "KEY_ENTER": + message = "52040817" + suffix; + break; + case "KEY_HOME": + message = "52040803" + suffix; + break; + case "KEY_BACK": + message = "52040804" + suffix; + break; + case "KEY_MENU": + message = "52040852" + suffix; + break; + case "KEY_PLAY": + message = "5204087E" + suffix; + break; + case "KEY_PAUSE": + message = "5204087F" + suffix; + break; + case "KEY_PLAYPAUSE": + message = "52040855" + suffix; + break; + case "KEY_STOP": + message = "52040856" + suffix; + break; + case "KEY_NEXT": + message = "52040857" + suffix; + break; + case "KEY_PREVIOUS": + message = "52040858" + suffix; + break; + case "KEY_REWIND": + message = "52040859" + suffix; + break; + case "KEY_FORWARD": + message = "5204085A" + suffix; + break; + case "KEY_POWER": + message = "5204081a" + suffix; + break; + case "KEY_VOLUP": + message = "52040818" + suffix; + break; + case "KEY_VOLDOWN": + message = "52040819" + suffix; + break; + case "KEY_MUTE": + message = "5204085b" + suffix; + break; + default: + logger.debug("Unknown Key {}", command); + return; + } + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage(message))); + } + } else if (CHANNEL_KEYCODE.equals(channelUID.getId())) { + if (command instanceof StringType) { + String shortCommand = command.toString().split("_")[0]; + int commandInt = Integer.parseInt(shortCommand, 10); + String suffix = ""; + if (commandInt > 255) { + suffix = "02"; + commandInt -= 256; + } else if (commandInt > 127) { + suffix = "01"; + } + + String key = Integer.toHexString(commandInt) + suffix; + + if ((key.length() % 2) == 1) { + key = "0" + key; + } + + key = "08" + key; + + if (command.toString().endsWith("_PRESS")) { + key = key + "1001"; + } else if (command.toString().endsWith("_RELEASE")) { + key = key + "1002"; + } else { + key = key + "1003"; + } + + String length = "0" + (key.length() / 2); + String message = "52" + length + key; + + logger.trace("Sending KEYCODE {} as {}", key, message); + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage(message))); + } + + } else if (CHANNEL_PINCODE.equals(channelUID.getId())) { + if (command instanceof StringType) { + try { + Certificate[] shimClientChain = this.shimClientChain; + Certificate[] shimServerChain = this.shimServerChain; + Certificate[] shimClientLocalChain = this.shimClientLocalChain; + if (config.mode.equals(DEFAULT_MODE)) { + if ((!isLoggedIn) && (command.toString().equals("REQUEST")) + && (childConnectionManager == null)) { + setStatus(false, "User Forced PIN Process"); + logger.debug("{} - User Forced PIN Process", handler.getThingID()); + disconnect(true); + startChildConnectionManager(config.port + 1, PIN_MODE); + try { + Thread.sleep(PIN_DELAY); + } catch (InterruptedException e) { + logger.trace("InterruptedException", e); + } + } + GoogleTVConnectionManager childConnectionManager = this.childConnectionManager; + if (childConnectionManager != null) { + childConnectionManager.handleCommand(channelUID, command); + } else { + logger.debug("{} - Child Connection Manager unavailable.", handler.getThingID()); + } + } else if ((config.mode.equals(PIN_MODE)) && (!config.shim)) { + if (!isLoggedIn) { + if (command.toString().equals("REQUEST")) { + sendCommand(new GoogleTVCommand( + GoogleTVRequest.encodeMessage(GoogleTVRequest.pinRequest(command.toString())))); + } else if (shimServerChain != null) { + this.pinHash = GoogleTVUtils.validatePIN(command.toString(), androidtvPKI.getCert(), + shimServerChain[0]); + sendCommand(new GoogleTVCommand( + GoogleTVRequest.encodeMessage(GoogleTVRequest.pinRequest(this.pinHash)))); + } + } + } else if ((config.mode.equals(PIN_MODE)) && (config.shim)) { + if ((shimClientChain != null) && (shimServerChain != null) && (shimClientLocalChain != null)) { + this.pinHash = GoogleTVUtils.validatePIN(command.toString(), androidtvPKI.getCert(), + shimServerChain[0]); + this.shimPinHash = GoogleTVUtils.validatePIN(command.toString(), shimClientChain[0], + shimClientLocalChain[0]); + } + } + } catch (CertificateException e) { + logger.trace("PIN CertificateException", e); + } + } + } else if (CHANNEL_POWER.equals(channelUID.getId())) { + if (command instanceof OnOffType) { + if ((power && command.equals(OnOffType.OFF)) || (!power && command.equals(OnOffType.ON))) { + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage("5204081a1003"))); + } + } else if (command instanceof StringType) { + if ((power && command.toString().equals("OFF")) || (!power && command.toString().equals("ON"))) { + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage("5204081a1003"))); + } + } + } else if (CHANNEL_MUTE.equals(channelUID.getId())) { + if (command instanceof OnOffType) { + if ((volMute && command.equals(OnOffType.OFF)) || (!volMute && command.equals(OnOffType.ON))) { + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage("5204085b1003"))); + } + } + } else if (CHANNEL_DEBUG.equals(channelUID.getId())) { + if (command instanceof StringType) { + if (command.toString().startsWith("RAW", 9)) { + String newCommand = command.toString().substring(13); + String message = GoogleTVRequest.encodeMessage(newCommand); + if (logger.isTraceEnabled()) { + logger.trace("Raw Message Decodes as: {}", GoogleTVRequest.decodeMessage(message)); + } + sendCommand(new GoogleTVCommand(message)); + } else if (command.toString().startsWith("MSG", 9)) { + String newCommand = command.toString().substring(13); + messageParser.handleMessage(newCommand); + } + } + } else if (CHANNEL_KEYBOARD.equals(channelUID.getId())) { + if (command instanceof StringType) { + String keyPress = ""; + for (int i = 0; i < command.toString().length(); i++) { + keyPress = "aa01071a0512031a01" + + GoogleTVRequest.decodeMessage(String.valueOf(command.toString().charAt(i))); + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage(keyPress))); + } + } + } else if (CHANNEL_PLAYER.equals(channelUID.getId())) { + String message = ""; + if (command == PlayPauseType.PAUSE || command == OnOffType.OFF) { + message = "5204087F1003"; + } else if (command == PlayPauseType.PLAY || command == OnOffType.ON) { + message = "5204087E1003"; + } else if (command == NextPreviousType.NEXT) { + message = "520408571003"; + } else if (command == NextPreviousType.PREVIOUS) { + message = "520408581003"; + } else if (command == RewindFastforwardType.FASTFORWARD) { + message = "5204085A1003"; + } else if (command == RewindFastforwardType.REWIND) { + message = "520408591003"; + } + sendCommand(new GoogleTVCommand(GoogleTVRequest.encodeMessage(message))); + } + } + + public void dispose() { + this.disposing = true; + + Future asyncInitializeTask = this.asyncInitializeTask; + if (asyncInitializeTask != null) { + asyncInitializeTask.cancel(true); // Interrupt async init task if it isn't done yet + } + Future shimAsyncInitializeTask = this.shimAsyncInitializeTask; + if (shimAsyncInitializeTask != null) { + shimAsyncInitializeTask.cancel(true); // Interrupt async init task if it isn't done yet + } + ScheduledFuture deviceHealthJob = this.deviceHealthJob; + if (deviceHealthJob != null) { + deviceHealthJob.cancel(true); + } + GoogleTVConnectionManager childConnectionManager = this.childConnectionManager; + if (childConnectionManager != null) { + childConnectionManager.dispose(); + } + disconnect(true); + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConstants.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConstants.java new file mode 100644 index 000000000..752a72df6 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVConstants.java @@ -0,0 +1,44 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.googletv; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * The {@link GoogleTVConstants} class defines common constants, which are + * used across the googletv protocol. + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class GoogleTVConstants { + + // List of all static String literals + public static final String DELIMITER_00 = "00"; + public static final String DELIMITER_01 = "01"; + public static final String DELIMITER_02 = "02"; + public static final String DELIMITER_08 = "08"; + public static final String DELIMITER_0A = "0a"; + public static final String DELIMITER_10 = "10"; + public static final String DELIMITER_12 = "12"; + public static final String DELIMITER_1A = "1a"; + public static final String DELIMITER_42 = "42"; + public static final String DELIMITER_92 = "92"; + public static final String DELIMITER_A2 = "a2"; + public static final String DELIMITER_C2 = "c2"; + + public static final String MESSAGE_POWEROFF = "c202020800"; + public static final String MESSAGE_POWERON = "c202020801"; + public static final String MESSAGE_PINSUCCESS = "080210c801ca02"; + public static final String HARD_DROP = "ffffffff"; +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVMessageParser.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVMessageParser.java new file mode 100644 index 000000000..a38f37106 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVMessageParser.java @@ -0,0 +1,336 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.googletv; + +import static org.openhab.binding.androidtv.internal.protocol.googletv.GoogleTVConstants.*; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * Class responsible for parsing incoming GoogleTV messages. Calls back to an object implementing the + * GoogleTVMessageParserCallbacks interface. + * + * Adapted from Lutron Leap binding + * + * @author Ben Rosenblum - Initial contribution + */ + +@NonNullByDefault +public class GoogleTVMessageParser { + private final Logger logger = LoggerFactory.getLogger(GoogleTVMessageParser.class); + + private final GoogleTVConnectionManager callback; + + public GoogleTVMessageParser(GoogleTVConnectionManager callback) { + this.callback = callback; + } + + public void handleMessage(String msg) { + if (msg.trim().isEmpty()) { + return; // Ignore empty lines + } + + String thingId = callback.getThingID(); + char[] charArray = msg.toCharArray(); + String lenString = "" + charArray[0] + charArray[1]; + int len = Integer.parseInt(lenString, 16); + msg = msg.substring(2); + charArray = msg.toCharArray(); + + logger.trace("{} - Received GoogleTV message - Length: {} Message: {}", thingId, len, msg); + + callback.validMessageReceived(); + + try { + if (msg.startsWith(DELIMITER_1A)) { + logger.warn("{} - GoogleTV Error Message: {}", thingId, msg); + } else if (msg.startsWith(DELIMITER_0A)) { + // First message on connection from GTV + // + // 0a 5b08 ff 041256 0a 11 534849454c4420416e64726f6964205456 12 06 4e5649444941 18 01 22 02 3131 2a + // ---------------------LEN-SHIELD Android TV--------------------LEN-NVIDIA---------LEN---LEN-Android + // 24 636f6d2e676f6f676c652e616e64726f69642e74762e72656d6f74652e73657276696365 32 + // LEN-com.google.android.tv.remote.service + // 0d 352e322e343733323534313333 + // LEN-5.2.473254133 + // + // 0a 5308 ff 04124e 0a 0c 42524156494120344b204742 12 04 536f6e79 18 01 22 01 39 2a + // ---------------------LEN-BRAVIA 4K GB---------------LEN-Sony-------LEN---LEN-Android Version + // 24 636f6d2e676f6f676c652e616e64726f69642e74762e72656d6f74652e73657276696365 32 + // 0d 352e322e343733323534313333 + // + // 0a 5408 ff 04124f 0a 0a 4368726f6d6563617374 12 06 476f6f676c65 18 01 22 02 3132 2a + // ---------------------LEN-Chromecast-------------LEN-Google---------LEN---LEN-Android Version + // 24 636f6d2e676f6f676c652e616e64726f69642e74762e72656d6f74652e73657276696365 32 + // 0d 352e322e343733323534313333 + // + // 0a 5708 ff 041252 0a 0d 4368726f6d6563617374204844 12 06 476f6f676c65 18 01 22 02 3132 2a + // ---------------------LEN-Chromecast HD----------------LEN-Google---------LEN---LEN-Android Version + // 24 636f6d2e676f6f676c652e616e64726f69642e74762e72656d6f74652e73657276696365 32 + // 0d352e322e343733323534313333 + + if (callback.getLoggedIn()) { + logger.warn("{} - Unexpected Login Message: {}", thingId, msg); + } else { + callback.sendCommand( + new GoogleTVCommand(GoogleTVRequest.encodeMessage(GoogleTVRequest.loginRequest(4)))); + } + + String st = ""; + int length = 0; + StringBuilder preambleSb = new StringBuilder(); + StringBuilder manufacturerSb = new StringBuilder(); + StringBuilder modelSb = new StringBuilder(); + StringBuilder androidVersionSb = new StringBuilder(); + StringBuilder remoteServerSb = new StringBuilder(); + StringBuilder remoteServerVersionSb = new StringBuilder(); + + int i = 0; + int current = 0; + + for (; i < 14; i++) { + preambleSb.append(charArray[i]); + } + + i += 2; // 0a delimiter + + st = "" + charArray[i] + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + current = i; + + for (; i < current + length; i++) { + modelSb.append(charArray[i]); + } + + i += 2; // 12 delimiter + + st = "" + charArray[i] + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + current = i; + + for (; i < current + length; i++) { + manufacturerSb.append(charArray[i]); + } + + i += 6; // 18 01 22 + + st = "" + charArray[i] + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + current = i; + + for (; i < current + length; i++) { + androidVersionSb.append(charArray[i]); + } + + i += 2; // 2a delimiter + + st = "" + charArray[i] + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + current = i; + + for (; i < current + length; i++) { + remoteServerSb.append(charArray[i]); + } + + i += 2; // 32 delimiter + + st = "" + charArray[i] + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + current = i; + + for (; i < current + length; i++) { + remoteServerVersionSb.append(charArray[i]); + } + + String preamble = preambleSb.toString(); + String model = GoogleTVRequest.encodeMessage(modelSb.toString()); + String manufacturer = GoogleTVRequest.encodeMessage(manufacturerSb.toString()); + String androidVersion = GoogleTVRequest.encodeMessage(androidVersionSb.toString()); + String remoteServer = GoogleTVRequest.encodeMessage(remoteServerSb.toString()); + String remoteServerVersion = GoogleTVRequest.encodeMessage(remoteServerVersionSb.toString()); + + logger.debug("{} - {} \"{}\" \"{}\" {} {} {}", thingId, preamble, model, manufacturer, androidVersion, + remoteServer, remoteServerVersion); + + callback.setModel(model); + callback.setManufacturer(manufacturer); + callback.setAndroidVersion(androidVersion); + callback.setRemoteServer(remoteServer); + callback.setRemoteServerVersion(remoteServerVersion); + + } else if (msg.startsWith(DELIMITER_12)) { + // Second message on connection from GTV + // Login successful + callback.sendCommand( + new GoogleTVCommand(GoogleTVRequest.encodeMessage(GoogleTVRequest.loginRequest(5)))); + logger.info("{} - Login Successful", thingId); + callback.setLoggedIn(true); + } else if (msg.startsWith(DELIMITER_92)) { + // Third message on connection from GTV + // Also sent on power state change (to ON only unless keypress triggers)i + // 9203 21 08 02 10 02 1a 11 534849454c4420416e64726f6964205456 20 02 2800 30 0f 38 0e 40 00 + // --------DD----DD----DD-LEN-SHIELD Android TV + // 9203 1e 08 9610 10 09 1a 0d 4368726f6d6563617374204844 20 02 2800 30 19 38 0a 40 00 + // --------DD------DD----DD-LEN-Chromecast HD + // 9203 1a 08 f304 10 09 1a 11 534849454c4420416e64726f6964205456 20 01 + // 9203 1a 08 8205 10 09 1a 11 534849454c4420416e64726f6964205456 20 01 + // --------DD------DD----DD-LEN-SHIELD Android TV + // + // VOLUME: + // ---------------DD----DD----DD-LEN-BRAVIA 4K GB------------DD---------DD-MAX---VOL---MUTE + // 00 --- 9203 1c 08 03 10 06 1a 0c 42524156494120344b204742 20 02 2800 30 64 38 00 40 00 + // 01 --- 9203 1c 08 03 10 06 1a 0c 42524156494120344b204742 20 02 2800 30 64 38 01 40 00 + // 100 -- 9203 1c 08 03 10 06 1a 0c 42524156494120344b204742 20 02 2800 30 64 38 64 40 00 + // MUTE - 9203 1c 08 03 10 06 1a 0c 42524156494120344b204742 20 02 2800 30 64 38 00 40 01 + + String st = ""; + int length = 0; + + StringBuilder preambleSb = new StringBuilder(); + StringBuilder modelSb = new StringBuilder(); + String volMax = ""; + String volCurr = ""; + String volMute = ""; + String audioMode = ""; + + int i = 0; + int current = 0; + + for (; i < 12; i++) { + preambleSb.append(charArray[i]); + } + + st = "" + charArray[i] + charArray[i + 1]; + do { + if (!DELIMITER_1A.equals(st)) { + preambleSb.append(st); + i += 2; + st = "" + charArray[i] + charArray[i + 1]; + } + } while (!DELIMITER_1A.equals(st)); + + i += 2; // 1a delimiter + + st = "" + charArray[i] + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + current = i; + + for (; i < current + length; i++) { + modelSb.append(charArray[i]); + } + + i += 2; // 20 delimiter + + st = "" + charArray[i] + charArray[i + 1]; + + audioMode = st; // 01 remote audio - 02 local audio + + if (DELIMITER_02.equals(st)) { + i += 2; // 02 longer message + i += 4; // Unknown 2800 message + i += 2; // 30 delimiter + volMax = "" + charArray[i] + charArray[i + 1]; + i += 4; // volMax + 38 delimiter + volCurr = "" + charArray[i] + charArray[i + 1]; + i += 4; // volCurr + 40 delimiter + volMute = "" + charArray[i] + charArray[i + 1]; + + callback.setVolMax(volMax); + callback.setVolCurr(volCurr); + callback.setVolMute(volMute); + } + + String preamble = preambleSb.toString(); + String model = GoogleTVRequest.encodeMessage(modelSb.toString()); + logger.debug("{} - Device Update: {} \"{}\" {} {} {} {}", thingId, preamble, model, audioMode, volMax, + volCurr, volMute); + callback.setAudioMode(audioMode); + + } else if (msg.startsWith(DELIMITER_08)) { + // PIN Process Messages. Only used on 6467. + if (msg.startsWith(MESSAGE_PINSUCCESS)) { + // PIN Process Successful + logger.debug("{} - PIN Process Successful!", thingId); + callback.finishPinProcess(); + } else { + // 080210c801a201081204080310061801 + // 080210c801fa0100 + logger.debug("{} - PIN Intermediary Message: {}", thingId, msg); + } + } else if (msg.startsWith(DELIMITER_C2)) { + // Power State + // c202020800 - OFF + // c202020801 - ON + if (MESSAGE_POWEROFF.equals(msg)) { + callback.setPower(false); + } else if (MESSAGE_POWERON.equals(msg)) { + callback.setPower(true); + } else { + logger.info("{} - Unknown power state received. {}", thingId, msg); + } + } else if (msg.startsWith(DELIMITER_42)) { + // Keepalive request + callback.sendKeepAlive(msg); + } else if (msg.startsWith(DELIMITER_A2)) { + // Current app name. Sent on keypress and power change. + // a201 21 0a 1f 62 1d 636f6d2e676f6f676c652e616e64726f69642e796f75747562652e7476 + // -----------------LEN-com.google.android.youtube.tv + // a201 21 0a 1f 62 1d 636f6d2e676f6f676c652e616e64726f69642e74766c61756e63686572 + // -----------------LEN-com.google.android.tvlauncher + // a201 14 0a 12 62 10 636f6d2e736f6e792e6474762e747678 + // -----------------LEN-com.sony.dtv.tvx + // a201 15 0a 13 62 11 636f6d2e6e6574666c69782e6e696e6a61 + // -----------------LEN-com.netflix.ninja + + StringBuilder preambleSb = new StringBuilder(); + StringBuilder appNameSb = new StringBuilder(); + int i = 0; + int current = 0; + + for (; i < 10; i++) { + preambleSb.append(charArray[i]); + } + + i += 2; // 62 delimiter + + String st = "" + charArray[i] + charArray[i + 1]; + int length = Integer.parseInt(st, 16) * 2; + i += 2; + current = i; + + for (; i < current + length; i++) { + appNameSb.append(charArray[i]); + } + + String preamble = preambleSb.toString(); + String appName = GoogleTVRequest.encodeMessage(appNameSb.toString()); + + logger.debug("{} - Current App: {} {}", thingId, preamble, appName); + callback.setCurrentApp(appName); + } else { + logger.info("{} - Unknown payload received. {} {}", thingId, len, msg); + } + } catch (Exception e) { + logger.debug("{} - Message Parser Exception on {}", thingId, msg); + logger.debug("Message Parser Caught Exception", e); + } + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVRequest.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVRequest.java new file mode 100644 index 000000000..0c976cd23 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVRequest.java @@ -0,0 +1,148 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.googletv; + +import static org.openhab.binding.androidtv.internal.AndroidTVBindingConstants.*; +import static org.openhab.binding.androidtv.internal.protocol.googletv.GoogleTVConstants.*; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * Contains static methods for constructing LEAP messages + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class GoogleTVRequest { + + public static String encodeMessage(String message) { + StringBuilder reply = new StringBuilder(); + char[] charArray = message.toCharArray(); + for (int i = 0; i < charArray.length; i = i + 2) { + String st = "" + charArray[i] + "" + charArray[i + 1]; + char ch = (char) Integer.parseInt(st, 16); + reply.append(ch); + } + return reply.toString(); + } + + public static String decodeMessage(String message) { + StringBuilder sb = new StringBuilder(); + char ch[] = message.toCharArray(); + for (int i = 0; i < ch.length; i++) { + String hexString = Integer.toHexString(ch[i]); + if (hexString.length() % 2 > 0) { + sb.append('0'); + } + sb.append(hexString); + } + return sb.toString(); + } + + public static String pinRequest(String pin) { + // OLD + if (PIN_REQUEST.equals(pin)) { + return loginRequest(3); + } else { + // 080210c801c202 22 0a 20 0e066c3d1c3a6686edb6b2648ff25fcb3f0bf9cc81deeee9fad1a26073645e17 + // 080210c801c202 22 0a 20 530bb7c7ba06069997285aff6e0106adfb19ab23c18a7422f5f643b35a6467b3 + // -------------------------SHA HASH OF PIN + + int length = pin.length() / 2; + String len1 = GoogleTVRequest.fixMessage(Integer.toHexString(length + 2)); + String len2 = GoogleTVRequest.fixMessage(Integer.toHexString(length)); + return "080210c801c202" + len1 + "0a" + len2 + pin; + } + } + + public static String loginRequest(int messageId) { + String message = ""; + if (messageId == 1) { + // Send app and device name + // 080210c801522d 0a 19 636f6d2e676f6f676c652e616e64726f69642e766964656f73 12 10 + // 73616d73756e6720534d2d4739393855 + // ------------------LEN com.google.android.videos----------------------------LEN samsung SM-G998U + message = "080210c801522d0a19636f6d2e676f6f676c652e616e64726f69642e766964656f73121073616d73756e6720534d2d4739393855"; + } else if (messageId == 2) { + // Unknown but required + // 080210c801a201 0e 0a 04 08031006 0a 04 08031004 1802 + // ---------------LEN---LEN------------LEN + message = "080210c801a2010e0a04080310060a04080310041802"; + } else if (messageId == 3) { + // Trigger PIN OSD + // ---------------LEN---LEN + // 080210c801a201 08 12 04 08031006 1801 + // 080210c801f201 08 0a 04 08031006 1001 + message = "080210c801f201080a04080310061001"; + } else if (messageId == 4) { + // 0a41087e123d0a 08 534d2d4739393855 12 07 73616d73756e67 18 01 22 02 3133 2a + // ---------------LEN--SM-G998U----------LEN--samsung--------- + // 19 636f6d2e676f6f676c652e616e64726f69642e766964656f73 32 + // LEN-com.google.android.videos---------------------------- + // 07 342e33382e3138 + // LEN-4.38.18 + // message = + // "0a41087e123d0a08534d2d4739393855120773616d73756e671801220231332a19636f6d2e676f6f676c652e616e64726f69642e766964656f733207342e33382e3138"; + + // 0a5708fe0412520a 08 534d2d4739393855 12 07 73616d73756e67 18 01 22 02 3133 2a + // -----------------LEN--SM-G998U----------LEN--samsung--------- + // 19 636f6d2e676f6f676c652e616e64726f69642e766964656f73 32 + // LEN-com.google.android.videos--------------------------- + // 1c 342e33392e3538342e3532393538383538332e372d72656c65617365 + // LEN-4.39.584.529588583.7-release + message = "0a5708fe0412520a08534d2d4739393855120773616d73756e671801220231332a19636f6d2e676f6f676c652e616e64726f69642e766964656f73321c342e33392e3538342e3532393538383538332e372d72656c65617365"; + } else if (messageId == 5) { + // Unknown. Sent after "1200" received + message = "1202087e"; + } + return message; + } + + public static String keepAlive(String request) { + // 42 07 08 01 10 e4f1 8d01 + // 4a 02 08 01 + + // 42 08 08 7f 10 b4 908a a819 + // 4a 02 08 7f + + // 42 09 08 8001 10 ed b78a a819 + // 4a 03 08 8001 + + char[] charArray = request.toCharArray(); + StringBuilder sb = new StringBuilder(); + sb.append(request); + sb.setLength(sb.toString().length() - 6); + String st = ""; + do { + int sbLen = sb.toString().length(); + st = "" + charArray[sbLen - 2] + charArray[sbLen - 1]; + if (!DELIMITER_10.equals(st)) { + sb.setLength(sbLen - 2); + } + } while (!DELIMITER_10.equals(st)); + sb.setLength(sb.toString().length() - 2); + + StringBuilder sbReply = new StringBuilder(); + for (int i = 4; i < sb.toString().length(); i++) { + sbReply.append(charArray[i]); + } + return "4a" + fixMessage(Integer.toHexString(sbReply.toString().length() / 2)) + sbReply.toString(); + } + + public static String fixMessage(String tempMsg) { + if (tempMsg.length() % 2 > 0) { + tempMsg = "0" + tempMsg; + } + return tempMsg; + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVUtils.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVUtils.java new file mode 100644 index 000000000..7747e9cd3 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/googletv/GoogleTVUtils.java @@ -0,0 +1,129 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.googletv; + +import java.math.BigInteger; +import java.security.MessageDigest; +import java.security.NoSuchAlgorithmException; +import java.security.PublicKey; +import java.security.cert.Certificate; +import java.security.interfaces.RSAPublicKey; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * GoogleTVCommand represents a GoogleTV protocol command + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class GoogleTVUtils { + private static final Logger LOGGER = LoggerFactory.getLogger(GoogleTVUtils.class); + + private static String processMag(final byte[] magnitude) { + final int length = magnitude.length; + if (length != 0) { + final BigInteger bigInteger = new BigInteger(1, magnitude); + final StringBuilder sb = new StringBuilder(); + sb.append("%0"); + sb.append(length + length); + sb.append("x"); + return String.format(sb.toString(), bigInteger); + } + return ""; + } + + private static final byte[] processDigestArray(final byte[] array) { + int n = 0; + int length; + while (true) { + length = array.length; + if (n >= length || array[n] != 0) { + break; + } + ++n; + } + final int n2 = length - n; + final byte[] array2 = new byte[n2]; + System.arraycopy(array, n, array2, 0, n2); + return array2; + } + + public static final byte[] processDigest(byte[] digest, Certificate clientCert, Certificate serverCert) { + final PublicKey clientPublicKey = clientCert.getPublicKey(); + final PublicKey serverPublicKey = serverCert.getPublicKey(); + processMag(digest); + if (clientPublicKey instanceof RSAPublicKey && serverPublicKey instanceof RSAPublicKey) { + final RSAPublicKey clientRSAPublicKey = (RSAPublicKey) clientPublicKey; + final RSAPublicKey serverRSAPublicKey = (RSAPublicKey) serverPublicKey; + try { + final MessageDigest instance = MessageDigest.getInstance("SHA-256"); + final byte[] byteArray1 = clientRSAPublicKey.getModulus().abs().toByteArray(); + final byte[] byteArray2 = clientRSAPublicKey.getPublicExponent().abs().toByteArray(); + final byte[] byteArray3 = serverRSAPublicKey.getModulus().abs().toByteArray(); + final byte[] byteArray4 = serverRSAPublicKey.getPublicExponent().abs().toByteArray(); + final byte[] r1 = processDigestArray(byteArray1); + final byte[] r2 = processDigestArray(byteArray2); + final byte[] r3 = processDigestArray(byteArray3); + final byte[] r4 = processDigestArray(byteArray4); + processMag(r1); + processMag(r2); + processMag(r3); + processMag(r4); + processMag(digest); + instance.update(r1); + instance.update(r2); + instance.update(r3); + instance.update(r4); + instance.update(digest); + digest = instance.digest(); + processMag(digest); + } catch (NoSuchAlgorithmException e) { + LOGGER.warn("NoSuchAlgorithmException Exception", e); + } + } + return digest; + } + + public static String byteArrayToString(byte[] array) { + StringBuilder sb = new StringBuilder(); + for (int i = 0; i < array.length; i++) { + sb.append((char) (array[i] & 0xFF)); + } + return sb.toString(); + } + + public static String validatePIN(String pin, Certificate clientCert, Certificate serverCert) { + char[] charArray = pin.toCharArray(); + + String s1 = "" + charArray[0] + charArray[1]; + String s2 = "" + charArray[2] + charArray[3]; + String s3 = "" + charArray[4] + charArray[5]; + int si1 = Integer.parseInt(s1, 16); + int si2 = Integer.parseInt(s2, 16); + int si3 = Integer.parseInt(s3, 16); + + byte[] sb123 = new byte[] { (byte) si1, (byte) si2, (byte) si3 }; + byte[] sb23 = new byte[] { (byte) si2, (byte) si3 }; + byte[] digest = processDigest(sb23, clientCert, serverCert); + String digestString = GoogleTVRequest.decodeMessage(byteArrayToString(digest)); + + byte[] validPinB = new byte[] { digest[0], (byte) si2, (byte) si3 }; + String validPin = GoogleTVRequest.decodeMessage(byteArrayToString(validPinB)); + LOGGER.trace("validatePIN {} {} {} {} {} {}", sb123, digest[0], sb23, validPinB, validPin, digestString); + + return digestString; + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVCommand.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVCommand.java new file mode 100644 index 000000000..a67f201c8 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVCommand.java @@ -0,0 +1,38 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.shieldtv; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * ShieldTVCommand represents a ShieldTV protocol command + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class ShieldTVCommand { + private String command; + + public ShieldTVCommand(String command) { + this.command = command; + } + + @Override + public String toString() { + return command; + } + + public boolean isEmpty() { + return command.isEmpty(); + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConfiguration.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConfiguration.java new file mode 100644 index 000000000..384761262 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConfiguration.java @@ -0,0 +1,34 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.shieldtv; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * The {@link ShieldTVConfiguration} class contains fields mapping thing configuration parameters. + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class ShieldTVConfiguration { + + public String ipAddress = ""; + public int port = 8987; + public int reconnect; + public int heartbeat; + public String keystoreFileName = ""; + public String keystorePassword = ""; + public int delay = 0; + public boolean shim; + public boolean shimNewKeys; +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConnectionManager.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConnectionManager.java new file mode 100644 index 000000000..3fed339d5 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConnectionManager.java @@ -0,0 +1,1219 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.shieldtv; + +import static org.openhab.binding.androidtv.internal.AndroidTVBindingConstants.*; +import static org.openhab.binding.androidtv.internal.protocol.shieldtv.ShieldTVConstants.*; + +import java.io.BufferedReader; +import java.io.BufferedWriter; +import java.io.File; +import java.io.IOException; +import java.io.InputStreamReader; +import java.io.InterruptedIOException; +import java.io.OutputStreamWriter; +import java.math.BigInteger; +import java.net.ConnectException; +import java.net.InetSocketAddress; +import java.net.NoRouteToHostException; +import java.net.ServerSocket; +import java.net.Socket; +import java.net.SocketAddress; +import java.net.SocketTimeoutException; +import java.net.UnknownHostException; +import java.nio.charset.StandardCharsets; +import java.security.GeneralSecurityException; +import java.security.NoSuchAlgorithmException; +import java.security.UnrecoverableKeyException; +import java.security.cert.Certificate; +import java.security.cert.X509Certificate; +import java.util.HashMap; +import java.util.Map; +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.Future; +import java.util.concurrent.LinkedBlockingQueue; +import java.util.concurrent.ScheduledExecutorService; +import java.util.concurrent.ScheduledFuture; +import java.util.concurrent.TimeUnit; + +import javax.net.ssl.KeyManagerFactory; +import javax.net.ssl.SSLContext; +import javax.net.ssl.SSLServerSocketFactory; +import javax.net.ssl.SSLSession; +import javax.net.ssl.SSLSocket; +import javax.net.ssl.SSLSocketFactory; +import javax.net.ssl.TrustManager; +import javax.net.ssl.X509TrustManager; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.binding.androidtv.internal.AndroidTVHandler; +import org.openhab.binding.androidtv.internal.utils.AndroidTVPKI; +import org.openhab.core.OpenHAB; +import org.openhab.core.library.types.StringType; +import org.openhab.core.thing.ChannelUID; +import org.openhab.core.types.Command; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * The {@link ShieldTVConnectionManager} is responsible for handling connections via the shieldtv protocol + * + * Significant portions reused from Lutron binding with permission from Bob A. + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class ShieldTVConnectionManager { + private static final int DEFAULT_RECONNECT_SECONDS = 60; + private static final int DEFAULT_HEARTBEAT_SECONDS = 5; + private static final long KEEPALIVE_TIMEOUT_SECONDS = 30; + private static final String DEFAULT_KEYSTORE_PASSWORD = "secret"; + private static final int DEFAULT_PORT = 8987; + + private final Logger logger = LoggerFactory.getLogger(ShieldTVConnectionManager.class); + + private ScheduledExecutorService scheduler; + + private final AndroidTVHandler handler; + private ShieldTVConfiguration config; + + private @NonNullByDefault({}) SSLSocketFactory sslSocketFactory; + private @Nullable SSLSocket sslSocket; + private @Nullable BufferedWriter writer; + private @Nullable BufferedReader reader; + + private @NonNullByDefault({}) SSLServerSocketFactory sslServerSocketFactory; + private @Nullable Socket shimServerSocket; + private @Nullable BufferedWriter shimWriter; + private @Nullable BufferedReader shimReader; + + private @NonNullByDefault({}) ShieldTVMessageParser messageParser; + + private final BlockingQueue sendQueue = new LinkedBlockingQueue<>(); + private final BlockingQueue shimQueue = new LinkedBlockingQueue<>(); + + private @Nullable Future asyncInitializeTask; + private @Nullable Future shimAsyncInitializeTask; + + private @Nullable Thread senderThread; + private @Nullable Thread readerThread; + private @Nullable Thread shimSenderThread; + private @Nullable Thread shimReaderThread; + + private @Nullable ScheduledFuture keepAliveJob; + private @Nullable ScheduledFuture keepAliveReconnectJob; + private @Nullable ScheduledFuture connectRetryJob; + private final Object keepAliveReconnectLock = new Object(); + private final Object connectionLock = new Object(); + private int periodicUpdate; + + private @Nullable ScheduledFuture deviceHealthJob; + private boolean isOnline = true; + + private StringBuffer sbReader = new StringBuffer(); + private StringBuffer sbShimReader = new StringBuffer(); + private String lastMsg = ""; + private String thisMsg = ""; + private boolean inMessage = false; + private String msgType = ""; + + private boolean disposing = false; + private boolean isLoggedIn = false; + private String statusMessage = ""; + + private String hostName = ""; + private String currentApp = ""; + private String deviceId = ""; + private String arch = ""; + + private AndroidTVPKI androidtvPKI = new AndroidTVPKI(); + private byte[] encryptionKey; + + private boolean appDBPopulated = false; + private Map appNameDB = new HashMap<>(); + private Map appURLDB = new HashMap<>(); + + public ShieldTVConnectionManager(AndroidTVHandler handler, ShieldTVConfiguration config) { + messageParser = new ShieldTVMessageParser(this); + this.config = config; + this.handler = handler; + this.scheduler = handler.getScheduler(); + this.encryptionKey = androidtvPKI.generateEncryptionKey(); + initialize(); + } + + public void setHostName(String hostName) { + this.hostName = hostName; + handler.setThingProperty("deviceName", hostName); + } + + public String getHostName() { + return hostName; + } + + public String getThingID() { + return handler.getThingID(); + } + + public void setDeviceID(String deviceId) { + this.deviceId = deviceId; + handler.setThingProperty("deviceID", deviceId); + } + + public String getDeviceID() { + return deviceId; + } + + public void setArch(String arch) { + this.arch = arch; + handler.setThingProperty("architectures", arch); + } + + public String getArch() { + return arch; + } + + public void setCurrentApp(String currentApp) { + this.currentApp = currentApp; + handler.updateChannelState(CHANNEL_APP, new StringType(currentApp)); + + if (this.appDBPopulated) { + String appName = ""; + String appURL = ""; + + if (appNameDB.get(currentApp) != null) { + appName = appNameDB.get(currentApp); + handler.updateChannelState(CHANNEL_APPNAME, new StringType(appName)); + } else { + logger.info("Unknown Android App: {}", currentApp); + handler.updateChannelState(CHANNEL_APPNAME, new StringType("")); + } + + if (appURLDB.get(currentApp) != null) { + appURL = appURLDB.get(currentApp); + handler.updateChannelState(CHANNEL_APPURL, new StringType(appURL)); + } else { + handler.updateChannelState(CHANNEL_APPURL, new StringType("")); + } + } + } + + public String getStatusMessage() { + return statusMessage; + } + + private void setStatus(boolean isLoggedIn) { + if (isLoggedIn) { + setStatus(isLoggedIn, "ONLINE"); + } else { + setStatus(isLoggedIn, "UNKNOWN"); + } + } + + private void setStatus(boolean isLoggedIn, String statusMessage) { + if ((this.isLoggedIn != isLoggedIn) || (!this.statusMessage.equals(statusMessage))) { + this.isLoggedIn = isLoggedIn; + this.statusMessage = statusMessage; + handler.checkThingStatus(); + } + } + + public String getCurrentApp() { + return currentApp; + } + + private void sendPeriodicUpdate() { + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("080b120308cd08"))); // Get Hostname + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("08f30712020805"))); // No Reply + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("08f10712020800"))); // Get App DB + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("08ec0712020806"))); // Get App + } + + public void setLoggedIn(boolean isLoggedIn) { + if (!this.isLoggedIn && isLoggedIn) { + sendPeriodicUpdate(); + } + + if (this.isLoggedIn != isLoggedIn) { + setStatus(isLoggedIn); + } + } + + public boolean getLoggedIn() { + return isLoggedIn; + } + + private boolean servicePing() { + int timeout = 500; + + SocketAddress socketAddress = new InetSocketAddress(config.ipAddress, config.port); + try (Socket socket = new Socket()) { + socket.connect(socketAddress, timeout); + return true; + } catch (ConnectException | SocketTimeoutException | NoRouteToHostException ignored) { + return false; + } catch (IOException ignored) { + // IOException is thrown by automatic close() of the socket. + // This should actually never return a value as we should return true above already + return true; + } + } + + private void checkHealth() { + boolean isOnline; + if (!isLoggedIn) { + isOnline = servicePing(); + } else { + isOnline = true; + } + logger.debug("{} - Device Health - Online: {} - Logged In: {}", handler.getThingID(), isOnline, isLoggedIn); + if (isOnline != this.isOnline) { + this.isOnline = isOnline; + if (isOnline) { + logger.debug("{} - Device is back online. Attempting reconnection.", handler.getThingID()); + reconnect(); + } + } + } + + public void setKeys(String privKey, String cert) { + try { + androidtvPKI.setKeys(privKey, encryptionKey, cert); + androidtvPKI.saveKeyStore(config.keystorePassword, encryptionKey); + } catch (GeneralSecurityException e) { + logger.debug("General security exception", e); + } catch (IOException e) { + logger.debug("IO Exception", e); + } catch (Exception e) { + logger.debug("General Exception", e); + } + } + + public void setAppDB(Map appNameDB, Map appURLDB) { + this.appNameDB = appNameDB; + this.appURLDB = appURLDB; + this.appDBPopulated = true; + logger.debug("{} - App DB Populated", handler.getThingID()); + logger.trace("{} - Handler appNameDB: {} appURLDB: {}", handler.getThingID(), this.appNameDB, this.appURLDB); + handler.updateCDP(CHANNEL_APP, this.appNameDB); + } + + private TrustManager[] defineNoOpTrustManager() { + return new TrustManager[] { new X509TrustManager() { + @Override + public void checkClientTrusted(final X509Certificate @Nullable [] chain, final @Nullable String authType) { + logger.debug("Assuming client certificate is valid"); + if (chain != null && logger.isTraceEnabled()) { + for (int cert = 0; cert < chain.length; cert++) { + logger.trace("Subject DN: {}", chain[cert].getSubjectX500Principal()); + logger.trace("Issuer DN: {}", chain[cert].getIssuerX500Principal()); + logger.trace("Serial number: {}", chain[cert].getSerialNumber()); + } + } + } + + @Override + public void checkServerTrusted(final X509Certificate @Nullable [] chain, final @Nullable String authType) { + logger.debug("Assuming server certificate is valid"); + if (chain != null && logger.isTraceEnabled()) { + for (int cert = 0; cert < chain.length; cert++) { + logger.trace("Subject DN: {}", chain[cert].getSubjectX500Principal()); + logger.trace("Issuer DN: {}", chain[cert].getIssuerX500Principal()); + logger.trace("Serial number: {}", chain[cert].getSerialNumber()); + } + } + } + + @Override + public X509Certificate @Nullable [] getAcceptedIssuers() { + return null; + } + } }; + } + + private void initialize() { + SSLContext sslContext; + + String folderName = OpenHAB.getUserDataFolder() + "/androidtv"; + File folder = new File(folderName); + + if (!folder.exists()) { + logger.debug("Creating directory {}", folderName); + folder.mkdirs(); + } + + config.port = (config.port > 0) ? config.port : DEFAULT_PORT; + config.reconnect = (config.reconnect > 0) ? config.reconnect : DEFAULT_RECONNECT_SECONDS; + config.heartbeat = (config.heartbeat > 0) ? config.heartbeat : DEFAULT_HEARTBEAT_SECONDS; + config.delay = (config.delay < 0) ? 0 : config.delay; + config.shim = (config.shim) ? true : false; + config.shimNewKeys = (config.shimNewKeys) ? true : false; + + config.keystoreFileName = (!config.keystoreFileName.equals("")) ? config.keystoreFileName + : folderName + "/shieldtv." + ((config.shim) ? "shim." : "") + handler.getThing().getUID().getId() + + ".keystore"; + config.keystorePassword = (!config.keystorePassword.equals("")) ? config.keystorePassword + : DEFAULT_KEYSTORE_PASSWORD; + + androidtvPKI.setKeystoreFileName(config.keystoreFileName); + androidtvPKI.setAlias("nvidia"); + + deviceHealthJob = scheduler.scheduleWithFixedDelay(this::checkHealth, config.heartbeat, config.heartbeat, + TimeUnit.SECONDS); + + try { + File keystoreFile = new File(config.keystoreFileName); + + if (!keystoreFile.exists() || config.shimNewKeys) { + androidtvPKI.generateNewKeyPair(encryptionKey); + androidtvPKI.saveKeyStore(config.keystorePassword, this.encryptionKey); + } else { + androidtvPKI.loadFromKeyStore(config.keystorePassword, this.encryptionKey); + } + + logger.trace("{} - Initializing SSL Context", handler.getThingID()); + KeyManagerFactory kmf = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm()); + kmf.init(androidtvPKI.getKeyStore(config.keystorePassword, this.encryptionKey), + config.keystorePassword.toCharArray()); + + TrustManager[] trustManagers = defineNoOpTrustManager(); + + sslContext = SSLContext.getInstance("TLS"); + sslContext.init(kmf.getKeyManagers(), trustManagers, null); + + sslSocketFactory = sslContext.getSocketFactory(); + if (!config.shim) { + asyncInitializeTask = scheduler.submit(this::connect); + } else { + shimAsyncInitializeTask = scheduler.submit(this::shimInitialize); + } + } catch (NoSuchAlgorithmException | IOException e) { + setStatus(false, "Error initializing keystore"); + logger.debug("Error initializing keystore", e); + } catch (UnrecoverableKeyException e) { + setStatus(false, "Key unrecoverable with supplied password"); + } catch (GeneralSecurityException e) { + logger.debug("General security exception", e); + } catch (Exception e) { + logger.debug("General exception", e); + } + } + + public void connect() { + synchronized (connectionLock) { + if (isOnline) { + try { + logger.debug("{} - Opening ShieldTV SSL connection to {}:{}", handler.getThingID(), + config.ipAddress, config.port); + SSLSocket sslSocket = (SSLSocket) sslSocketFactory.createSocket(config.ipAddress, config.port); + sslSocket.startHandshake(); + writer = new BufferedWriter( + new OutputStreamWriter(sslSocket.getOutputStream(), StandardCharsets.ISO_8859_1)); + reader = new BufferedReader( + new InputStreamReader(sslSocket.getInputStream(), StandardCharsets.ISO_8859_1)); + this.sslSocket = sslSocket; + } catch (UnknownHostException e) { + setStatus(false, "Unknown host"); + return; + } catch (IllegalArgumentException e) { + // port out of valid range + setStatus(false, "Invalid port number"); + return; + } catch (InterruptedIOException e) { + logger.debug("Interrupted while establishing ShieldTV connection"); + Thread.currentThread().interrupt(); + return; + } catch (IOException e) { + setStatus(false, "Error opening ShieldTV SSL connection. Check log."); + logger.info("{} - Error opening ShieldTV SSL connection to {}:{} {}", handler.getThingID(), + config.ipAddress, config.port, e.getMessage()); + disconnect(false); + scheduleConnectRetry(config.reconnect); // Possibly a temporary problem. Try again later. + return; + } + + setStatus(false, "Initializing"); + + Thread readerThread = new Thread(this::readerThreadJob, "ShieldTV reader " + handler.getThingID()); + readerThread.setDaemon(true); + readerThread.start(); + this.readerThread = readerThread; + + Thread senderThread = new Thread(this::senderThreadJob, "ShieldTV sender " + handler.getThingID()); + senderThread.setDaemon(true); + senderThread.start(); + this.senderThread = senderThread; + + if (!config.shim) { + this.periodicUpdate = 20; + logger.debug("{} - Starting ShieldTV keepalive job with interval {}", handler.getThingID(), + config.heartbeat); + keepAliveJob = scheduler.scheduleWithFixedDelay(this::sendKeepAlive, config.heartbeat, + config.heartbeat, TimeUnit.SECONDS); + + String login = ShieldTVRequest.encodeMessage(ShieldTVRequest.loginRequest()); + sendCommand(new ShieldTVCommand(login)); + } + } else { + scheduleConnectRetry(config.reconnect); // Possibly a temporary problem. Try again later. + } + } + } + + public void shimInitialize() { + synchronized (connectionLock) { + AndroidTVPKI shimPKI = new AndroidTVPKI(); + byte[] shimEncryptionKey = shimPKI.generateEncryptionKey(); + SSLContext sslContext; + + try { + shimPKI.generateNewKeyPair(shimEncryptionKey); + KeyManagerFactory kmf = KeyManagerFactory.getInstance(KeyManagerFactory.getDefaultAlgorithm()); + kmf.init(shimPKI.getKeyStore(config.keystorePassword, shimEncryptionKey), + config.keystorePassword.toCharArray()); + TrustManager[] trustManagers = defineNoOpTrustManager(); + sslContext = SSLContext.getInstance("TLS"); + sslContext.init(kmf.getKeyManagers(), trustManagers, null); + this.sslServerSocketFactory = sslContext.getServerSocketFactory(); + + logger.debug("{} - Opening ShieldTV shim on port {}", handler.getThingID(), config.port); + ServerSocket sslServerSocket = this.sslServerSocketFactory.createServerSocket(config.port); + + while (true) { + logger.debug("{} - Waiting for shim connection...", handler.getThingID()); + Socket serverSocket = sslServerSocket.accept(); + disconnect(false); + connect(); + SSLSession session = ((SSLSocket) serverSocket).getSession(); + Certificate[] cchain2 = session.getLocalCertificates(); + for (int i = 0; i < cchain2.length; i++) { + logger.trace("Connection from: {}", ((X509Certificate) cchain2[i]).getSubjectX500Principal()); + } + + logger.trace("Peer host is {}", session.getPeerHost()); + logger.trace("Cipher is {}", session.getCipherSuite()); + logger.trace("Protocol is {}", session.getProtocol()); + logger.trace("ID is {}", new BigInteger(session.getId())); + logger.trace("Session created in {}", session.getCreationTime()); + logger.trace("Session accessed in {}", session.getLastAccessedTime()); + + shimWriter = new BufferedWriter( + new OutputStreamWriter(serverSocket.getOutputStream(), StandardCharsets.ISO_8859_1)); + shimReader = new BufferedReader( + new InputStreamReader(serverSocket.getInputStream(), StandardCharsets.ISO_8859_1)); + this.shimServerSocket = serverSocket; + + Thread readerThread = new Thread(this::shimReaderThreadJob, + "ShieldTV shim reader " + handler.getThingID()); + readerThread.setDaemon(true); + readerThread.start(); + this.shimReaderThread = readerThread; + + Thread senderThread = new Thread(this::shimSenderThreadJob, + "ShieldTV shim sender" + handler.getThingID()); + senderThread.setDaemon(true); + senderThread.start(); + this.shimSenderThread = senderThread; + } + } catch (Exception e) { + logger.trace("Shim initalization exception", e); + return; + } + } + } + + private void scheduleConnectRetry(long waitSeconds) { + logger.trace("{} - Scheduling ShieldTV connection retry in {} seconds", handler.getThingID(), waitSeconds); + connectRetryJob = scheduler.schedule(this::connect, waitSeconds, TimeUnit.SECONDS); + } + + /** + * Disconnect from bridge, cancel retry and keepalive jobs, stop reader and writer threads, and + * clean up. + * + * @param interruptAll Set if reconnect task should be interrupted if running. Should be false when calling from + * connect or reconnect, and true when calling from dispose. + */ + private void disconnect(boolean interruptAll) { + synchronized (connectionLock) { + logger.debug("{} - Disconnecting ShieldTV", handler.getThingID()); + + this.isLoggedIn = false; + + ScheduledFuture connectRetryJob = this.connectRetryJob; + if (connectRetryJob != null) { + connectRetryJob.cancel(true); + } + ScheduledFuture keepAliveJob = this.keepAliveJob; + if (keepAliveJob != null) { + keepAliveJob.cancel(true); + } + + reconnectTaskCancel(interruptAll); // May be called from keepAliveReconnectJob thread + + Thread senderThread = this.senderThread; + if (senderThread != null && senderThread.isAlive()) { + senderThread.interrupt(); + } + + Thread readerThread = this.readerThread; + if (readerThread != null && readerThread.isAlive()) { + readerThread.interrupt(); + } + + Thread shimSenderThread = this.shimSenderThread; + if (shimSenderThread != null && shimSenderThread.isAlive()) { + shimSenderThread.interrupt(); + } + + Thread shimReaderThread = this.shimReaderThread; + if (shimReaderThread != null && shimReaderThread.isAlive()) { + shimReaderThread.interrupt(); + } + + SSLSocket sslSocket = this.sslSocket; + if (sslSocket != null) { + try { + sslSocket.close(); + } catch (IOException e) { + logger.debug("Error closing ShieldTV SSL socket: {}", e.getMessage()); + } + this.sslSocket = null; + } + BufferedReader reader = this.reader; + if (reader != null) { + try { + reader.close(); + } catch (IOException e) { + logger.debug("Error closing reader: {}", e.getMessage()); + } + } + BufferedWriter writer = this.writer; + if (writer != null) { + try { + writer.close(); + } catch (IOException e) { + logger.debug("Error closing writer: {}", e.getMessage()); + } + } + + Socket shimServerSocket = this.shimServerSocket; + if (shimServerSocket != null) { + try { + shimServerSocket.close(); + } catch (IOException e) { + logger.debug("Error closing ShieldTV SSL socket: {}", e.getMessage()); + } + this.shimServerSocket = null; + } + BufferedReader shimReader = this.shimReader; + if (shimReader != null) { + try { + shimReader.close(); + } catch (IOException e) { + logger.debug("Error closing shimReader: {}", e.getMessage()); + } + } + BufferedWriter shimWriter = this.shimWriter; + if (shimWriter != null) { + try { + shimWriter.close(); + } catch (IOException e) { + logger.debug("Error closing shimWriter: {}", e.getMessage()); + } + } + } + } + + private void reconnect() { + synchronized (connectionLock) { + if (!this.disposing) { + logger.debug("{} - Attempting to reconnect to the ShieldTV", handler.getThingID()); + setStatus(false, "reconnecting"); + disconnect(false); + connect(); + } + } + } + + /** + * Method executed by the message sender thread (senderThread) + */ + private void senderThreadJob() { + logger.debug("{} - Command sender thread started", handler.getThingID()); + try { + while (!Thread.currentThread().isInterrupted() && writer != null) { + ShieldTVCommand command = sendQueue.take(); + + try { + BufferedWriter writer = this.writer; + if (writer != null) { + logger.trace("{} - Raw ShieldTV command decodes as: {}", handler.getThingID(), + ShieldTVRequest.decodeMessage(command.toString())); + writer.write(command.toString()); + writer.flush(); + } + } catch (InterruptedIOException e) { + logger.debug("Interrupted while sending to ShieldTV"); + setStatus(false, "Interrupted"); + break; // exit loop and terminate thread + } catch (IOException e) { + logger.warn("{} - Communication error, will try to reconnect ShieldTV. Error: {}", + handler.getThingID(), e.getMessage()); + setStatus(false, "Communication error, will try to reconnect"); + sendQueue.add(command); // Requeue command + this.isLoggedIn = false; + reconnect(); + break; // reconnect() will start a new thread; terminate this one + } + if (config.delay > 0) { + Thread.sleep(config.delay); // introduce delay to throttle send rate + } + } + } catch (InterruptedException e) { + Thread.currentThread().interrupt(); + } finally { + logger.debug("{} - Command sender thread exiting", handler.getThingID()); + } + } + + private void shimSenderThreadJob() { + logger.debug("Shim sender thread started"); + try { + while (!Thread.currentThread().isInterrupted() && shimWriter != null) { + ShieldTVCommand command = shimQueue.take(); + + try { + BufferedWriter writer = this.shimWriter; + if (writer != null) { + logger.trace("Shim received from shield: {}", + ShieldTVRequest.decodeMessage(command.toString())); + writer.write(command.toString()); + writer.flush(); + } + } catch (InterruptedIOException e) { + logger.debug("Shim interrupted while sending."); + break; // exit loop and terminate thread + } catch (IOException e) { + logger.warn("Shim communication error. Error: {}", e.getMessage()); + break; // reconnect() will start a new thread; terminate this one + } + } + } catch (InterruptedException e) { + Thread.currentThread().interrupt(); + } finally { + logger.debug("Command sender thread exiting"); + } + } + + private void flushReader() { + if (!inMessage && (sbReader.length() > 0)) { + sbReader.setLength(sbReader.length() - 2); + messageParser.handleMessage(sbReader.toString()); + if (config.shim) { + sendShim(new ShieldTVCommand(ShieldTVRequest.encodeMessage(sbReader.toString()))); + } + sbReader.setLength(0); + sbReader.append(lastMsg); + } + sbReader.append(thisMsg); + lastMsg = thisMsg; + } + + private void finishReaderMessage() { + sbReader.append(thisMsg); + lastMsg = ""; + inMessage = false; + messageParser.handleMessage(sbReader.toString()); + if (config.shim) { + sendShim(new ShieldTVCommand(ShieldTVRequest.encodeMessage(sbReader.toString()))); + } + sbReader.setLength(0); + } + + private String fixMessage(String tempMsg) { + if (tempMsg.length() % 2 > 0) { + tempMsg = "0" + tempMsg; + } + return tempMsg; + } + + /** + * Method executed by the message reader thread (readerThread) + */ + private void readerThreadJob() { + logger.debug("{} - Message reader thread started", handler.getThingID()); + try { + BufferedReader reader = this.reader; + while (!Thread.interrupted() && reader != null) { + thisMsg = fixMessage(Integer.toHexString(reader.read())); + if (HARD_DROP.equals(thisMsg)) { + // Shield has crashed the connection. Disconnect hard. + logger.debug("{} - readerThreadJob received ffffffff. Disconnecting hard.", handler.getThingID()); + this.isLoggedIn = false; + reconnect(); + break; + } + if (DELIMITER_08.equals(lastMsg) && !inMessage) { + flushReader(); + inMessage = true; + msgType = thisMsg; + } else if (DELIMITER_18.equals(lastMsg) && thisMsg.equals(msgType) && inMessage) { + if (!msgType.startsWith(DELIMITER_0)) { + sbReader.append(thisMsg); + thisMsg = fixMessage(Integer.toHexString(reader.read())); + } + finishReaderMessage(); + } else if (DELIMITER_00.equals(msgType) && (sbReader.toString().length() == 16)) { + // keepalive messages don't have delimiters but are always 18 in length + finishReaderMessage(); + } else { + sbReader.append(thisMsg); + lastMsg = thisMsg; + } + } + } catch (InterruptedIOException e) { + logger.debug("Interrupted while reading"); + setStatus(false, "Interrupted"); + } catch (IOException e) { + logger.debug("I/O error while reading from stream: {}", e.getMessage()); + setStatus(false, "I/O Error"); + } catch (RuntimeException e) { + logger.warn("Runtime exception in reader thread", e); + setStatus(false, "Runtime exception"); + } finally { + logger.debug("{} - Message reader thread exiting", handler.getThingID()); + } + } + + private void shimReaderThreadJob() { + logger.debug("Shim reader thread started"); + String thisShimMsg = ""; + int thisShimRawMsg = 0; + int payloadRemain = 0; + int payloadBlock = 0; + String thisShimMsgType = ""; + boolean inShimMessage = false; + try { + BufferedReader reader = this.shimReader; + while (!Thread.interrupted() && reader != null) { + thisShimRawMsg = reader.read(); + thisShimMsg = fixMessage(Integer.toHexString(thisShimRawMsg)); + if (HARD_DROP.equals(thisShimMsg)) { + disconnect(false); + break; + } + if (!inShimMessage) { + // Beginning of payload + sbShimReader.setLength(0); + sbShimReader.append(thisShimMsg); + inShimMessage = true; + payloadBlock++; + } else if ((payloadBlock == 1) && (DELIMITER_00.equals(thisShimMsg))) { + sbShimReader.append(thisShimMsg); + payloadRemain = 8; + thisShimMsgType = thisShimMsg; + while (payloadRemain > 1) { + thisShimMsg = fixMessage(Integer.toHexString(reader.read())); + sbShimReader.append(thisShimMsg); + payloadRemain--; + payloadBlock++; + } + payloadRemain--; + payloadBlock++; + } else if ((payloadBlock == 1) + && (thisShimMsg.startsWith(DELIMITER_F1) || thisShimMsg.startsWith(DELIMITER_F3))) { + sbShimReader.append(thisShimMsg); + payloadRemain = 6; + thisShimMsgType = thisShimMsg; + while (payloadRemain > 1) { + thisShimMsg = fixMessage(Integer.toHexString(reader.read())); + sbShimReader.append(thisShimMsg); + payloadRemain--; + payloadBlock++; + } + payloadRemain--; + payloadBlock++; + } else if (payloadBlock == 1) { + thisShimMsgType = thisShimMsg; + sbShimReader.append(thisShimMsg); + payloadBlock++; + } else if (payloadBlock == 2) { + sbShimReader.append(thisShimMsg); + payloadBlock++; + } else if (payloadBlock == 3) { + // Length of remainder of packet + payloadRemain = thisShimRawMsg; + sbShimReader.append(thisShimMsg); + payloadBlock++; + } else if (payloadBlock == 4) { + sbShimReader.append(thisShimMsg); + logger.trace("PB4 SSR {} TSMT {} TSM {} PR {}", sbShimReader.toString(), thisShimMsgType, + thisShimMsg, payloadRemain); + if (DELIMITER_E9.equals(thisShimMsgType) || DELIMITER_F0.equals(thisShimMsgType) + || DELIMITER_EC.equals(thisShimMsgType)) { + payloadRemain = thisShimRawMsg + 1; + } + while (payloadRemain > 1) { + thisShimMsg = fixMessage(Integer.toHexString(reader.read())); + sbShimReader.append(thisShimMsg); + payloadRemain--; + payloadBlock++; + } + payloadRemain--; + payloadBlock++; + } + + if ((payloadBlock > 5) && (payloadRemain == 0)) { + logger.trace("Shim sending to shield: {}", sbShimReader.toString()); + sendQueue.add(new ShieldTVCommand(ShieldTVRequest.encodeMessage(sbShimReader.toString()))); + inShimMessage = false; + payloadBlock = 0; + payloadRemain = 0; + sbShimReader.setLength(0); + } + } + } catch (InterruptedIOException e) { + logger.debug("Interrupted while reading"); + setStatus(false, "Interrupted"); + } catch (IOException e) { + logger.debug("I/O error while reading from stream: {}", e.getMessage()); + setStatus(false, "I/O Error"); + } catch (RuntimeException e) { + logger.warn("Runtime exception in reader thread", e); + setStatus(false, "Runtime exception"); + } finally { + logger.debug("Message reader thread exiting"); + } + } + + private void sendKeepAlive() { + logger.trace("{} - Sending ShieldTV keepalive query", handler.getThingID()); + String keepalive = ShieldTVRequest.encodeMessage(ShieldTVRequest.keepAlive()); + sendCommand(new ShieldTVCommand(keepalive)); + if (isLoggedIn) { + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("08ec0712020806"))); // Get App + if (this.periodicUpdate <= 1) { + sendPeriodicUpdate(); + this.periodicUpdate = 20; + } else { + periodicUpdate--; + } + } + reconnectTaskSchedule(); + } + + /** + * Schedules the reconnect task keepAliveReconnectJob to execute in KEEPALIVE_TIMEOUT_SECONDS. This should + * be + * cancelled by calling reconnectTaskCancel() if a valid response is received from the bridge. + */ + private void reconnectTaskSchedule() { + synchronized (keepAliveReconnectLock) { + keepAliveReconnectJob = scheduler.schedule(this::keepAliveTimeoutExpired, KEEPALIVE_TIMEOUT_SECONDS, + TimeUnit.SECONDS); + } + } + + /** + * Cancels the reconnect task keepAliveReconnectJob. + */ + private void reconnectTaskCancel(boolean interrupt) { + synchronized (keepAliveReconnectLock) { + ScheduledFuture keepAliveReconnectJob = this.keepAliveReconnectJob; + if (keepAliveReconnectJob != null) { + logger.trace("{} - Canceling ShieldTV scheduled reconnect job.", handler.getThingID()); + keepAliveReconnectJob.cancel(interrupt); + this.keepAliveReconnectJob = null; + } + } + } + + /** + * Executed by keepAliveReconnectJob if it is not cancelled by the LEAP message parser calling + * validMessageReceived() which in turn calls reconnectTaskCancel(). + */ + private void keepAliveTimeoutExpired() { + logger.debug("{} - ShieldTV keepalive response timeout expired. Initiating reconnect.", handler.getThingID()); + reconnect(); + } + + public void validMessageReceived() { + reconnectTaskCancel(true); // Got a good message, so cancel reconnect task. + } + + public void sendCommand(ShieldTVCommand command) { + if ((!config.shim) && (!command.isEmpty())) { + sendQueue.add(command); + } + } + + public void sendShim(ShieldTVCommand command) { + if (!command.isEmpty()) { + shimQueue.add(command); + } + } + + public void handleCommand(ChannelUID channelUID, Command command) { + logger.debug("{} - Command received: {}", handler.getThingID(), channelUID.getId()); + + if (CHANNEL_KEYPRESS.equals(channelUID.getId())) { + if (command instanceof StringType) { + switch (command.toString()) { + case "KEY_UP": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202ce01"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202ce01"))); + break; + case "KEY_DOWN": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202d801"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202d801"))); + break; + case "KEY_RIGHT": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202d401"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202d401"))); + break; + case "KEY_LEFT": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202d201"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202d201"))); + break; + case "KEY_ENTER": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202c205"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202c205"))); + break; + case "KEY_HOME": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202d802"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202d802"))); + break; + case "KEY_BACK": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202bc02"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202bc02"))); + break; + case "KEY_MENU": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a280132029602"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a280232029602"))); + break; + case "KEY_PLAYPAUSE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202F604"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202F604"))); + break; + case "KEY_REWIND": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202D002"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202D002"))); + break; + case "KEY_FORWARD": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202A003"))); + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202A003"))); + break; + case "KEY_UP_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202ce01"))); + break; + case "KEY_DOWN_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202d801"))); + break; + case "KEY_RIGHT_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202d401"))); + break; + case "KEY_LEFT_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202d201"))); + break; + case "KEY_ENTER_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202c205"))); + break; + case "KEY_HOME_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202d802"))); + break; + case "KEY_BACK_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202bc02"))); + break; + case "KEY_MENU_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a280132029602"))); + break; + case "KEY_PLAYPAUSE_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202F604"))); + break; + case "KEY_REWIND_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202D002"))); + break; + case "KEY_FORWARD_PRESS": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28013202A003"))); + break; + case "KEY_UP_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202ce01"))); + break; + case "KEY_DOWN_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202d801"))); + break; + case "KEY_RIGHT_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202d401"))); + break; + case "KEY_LEFT_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202d201"))); + break; + case "KEY_ENTER_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202c205"))); + break; + case "KEY_HOME_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202d802"))); + break; + case "KEY_BACK_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202bc02"))); + break; + case "KEY_MENU_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a280232029602"))); + break; + case "KEY_PLAYPAUSE_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202F604"))); + break; + case "KEY_REWIND_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202D002"))); + break; + case "KEY_FORWARD_RELEASE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08e907120c08141001200a28023202A003"))); + break; + case "KEY_POWER": + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("08e907120808141005201e401e"))); + break; + case "KEY_POWERON": + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("08e907120808141005201e4010"))); + break; + case "KEY_GOOGLE": + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("08e907120808141005201e401f"))); + break; + case "KEY_VOLUP": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08f007120c08031208080110031a020102"))); + break; + case "KEY_VOLDOWN": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08f007120c08031208080110011a020102"))); + break; + case "KEY_MUTE": + sendCommand(new ShieldTVCommand( + ShieldTVRequest.encodeMessage("08f007120c08031208080110021a020102"))); + break; + case "KEY_SUBMIT": + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("08e9071209081410012001320138"))); + break; + } + if (command.toString().length() == 5) { + // Account for KEY_(ASCII Character) + String keyPress = "08ec07120708011201" + + ShieldTVRequest.decodeMessage(new String("" + command.toString().charAt(4))) + "1801"; + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage(keyPress))); + } else { + logger.trace("Unknown Keypress: {}", command.toString()); + } + } + } else if (CHANNEL_PINCODE.equals(channelUID.getId())) { + if (command instanceof StringType) { + if (!isLoggedIn) { + // Do PIN for shieldtv protocol + logger.debug("{} - ShieldTV PIN Process Started", handler.getThingID()); + String pin = ShieldTVRequest.pinRequest(command.toString()); + String message = ShieldTVRequest.encodeMessage(pin); + sendCommand(new ShieldTVCommand(message)); + } + } + } else if (CHANNEL_DEBUG.equals(channelUID.getId())) { + if (command instanceof StringType) { + if (command.toString().startsWith("RAW", 9)) { + String newCommand = command.toString().substring(13); + String message = ShieldTVRequest.encodeMessage(newCommand); + if (logger.isTraceEnabled()) { + logger.trace("Raw Message Decodes as: {}", ShieldTVRequest.decodeMessage(message)); + } + sendCommand(new ShieldTVCommand(message)); + } else if (command.toString().startsWith("MSG", 9)) { + String newCommand = command.toString().substring(13); + messageParser.handleMessage(newCommand); + } + } + } else if (CHANNEL_APP.equals(channelUID.getId())) { + if (command instanceof StringType) { + String message = ShieldTVRequest.encodeMessage(ShieldTVRequest.startApp(command.toString())); + sendCommand(new ShieldTVCommand(message)); + } + } else if (CHANNEL_KEYBOARD.equals(channelUID.getId())) { + if (command instanceof StringType) { + String entry = ShieldTVRequest.keyboardEntry(command.toString()); + logger.trace("Keyboard Entry {}", entry); + String message = ShieldTVRequest.encodeMessage(entry); + sendCommand(new ShieldTVCommand(message)); + sendCommand(new ShieldTVCommand(ShieldTVRequest.encodeMessage("08e9071209081410012001320138"))); + } + } + } + + public void dispose() { + this.disposing = true; + + Future asyncInitializeTask = this.asyncInitializeTask; + if (asyncInitializeTask != null) { + asyncInitializeTask.cancel(true); // Interrupt async init task if it isn't done yet + } + Future shimAsyncInitializeTask = this.shimAsyncInitializeTask; + if (shimAsyncInitializeTask != null) { + shimAsyncInitializeTask.cancel(true); // Interrupt async init task if it isn't done yet + } + ScheduledFuture deviceHealthJob = this.deviceHealthJob; + if (deviceHealthJob != null) { + deviceHealthJob.cancel(true); + } + disconnect(true); + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConstants.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConstants.java new file mode 100644 index 000000000..59615822b --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVConstants.java @@ -0,0 +1,61 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.shieldtv; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * The {@link ShieldTVConstants} class defines common constants, which are + * used across the shieldtv protocol. + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class ShieldTVConstants { + + // List of all static String literals + public static final String DELIMITER_0 = "0"; + public static final String DELIMITER_00 = "00"; + public static final String DELIMITER_08 = "08"; + public static final String DELIMITER_0A = "0a"; + public static final String DELIMITER_12 = "12"; + public static final String DELIMITER_18 = "18"; + public static final String DELIMITER_22 = "22"; + public static final String DELIMITER_2A = "2a"; + public static final String DELIMITER_E9 = "e9"; + public static final String DELIMITER_EC = "ec"; + public static final String DELIMITER_F0 = "f0"; + public static final String DELIMITER_F1 = "f1"; + public static final String DELIMITER_F3 = "f3"; + + public static final String HARD_DROP = "ffffffff"; + + public static final String APP_START_SUCCESS = "08f1071202080318f107"; + public static final String APP_START_FAILED = "08f107120608031202080118f107"; + public static final String KEEPALIVE_REPLY = "080028fae0a6c0d130"; + public static final String TIMEOUT = "080a121108b510120c0804120854696d65206f7574180a"; + + public static final String MESSAGE_LOWPRIV = "080a12"; + public static final String MESSAGE_HOSTNAME = "080b12"; + public static final String MESSAGE_APPDB = "08f10712"; + public static final String MESSAGE_GOOD_COMMAND = "08f30712"; + public static final String MESSAGE_PINSTART = "0308cf08"; + public static final String MESSAGE_CERT_COMING = "20"; + public static final String MESSAGE_SUCCESS = "08f007"; + public static final String MESSAGE_APP_SUCCESS = "08ec07"; + public static final String MESSAGE_APP_GET_SUCCESS = "0803"; + public static final String MESSAGE_APP_CURRENT = "0807"; + public static final String MESSAGE_SHORTNAME = "08e807"; + public static final String MESSAGE_CERT = "08b510"; + public static final String MESSAGE_CERT_PAYLOAD = "0753756363657373"; +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVMessageParser.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVMessageParser.java new file mode 100644 index 000000000..2707c22a2 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVMessageParser.java @@ -0,0 +1,445 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.shieldtv; + +import static org.openhab.binding.androidtv.internal.protocol.shieldtv.ShieldTVConstants.*; + +import java.util.ArrayList; +import java.util.Base64; +import java.util.Collections; +import java.util.HashMap; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; +import java.util.Map.Entry; + +import javax.xml.bind.DatatypeConverter; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * Class responsible for parsing incoming ShieldTV messages. Calls back to an object implementing the + * ShieldTVMessageParserCallbacks interface. + * + * Adapted from Lutron Leap binding + * + * @author Ben Rosenblum - Initial contribution + */ + +@NonNullByDefault +public class ShieldTVMessageParser { + private final Logger logger = LoggerFactory.getLogger(ShieldTVMessageParser.class); + + private final ShieldTVConnectionManager callback; + + public ShieldTVMessageParser(ShieldTVConnectionManager callback) { + this.callback = callback; + } + + public void handleMessage(String msg) { + if (msg.trim().isEmpty()) { + return; // Ignore empty lines + } + + String thingId = callback.getThingID(); + String hostName = callback.getHostName(); + logger.trace("{} - Received ShieldTV message from: {} - Message: {}", thingId, hostName, msg); + + callback.validMessageReceived(); + + char[] charArray = msg.toCharArray(); + + try { + // All lengths are little endian when larger than 0xff + if (msg.startsWith(MESSAGE_LOWPRIV) && msg.startsWith(MESSAGE_SHORTNAME, 8)) { + // Pre-login Hostname of Shield Replied + // 080a 12 1408e807 12 0f08e807 12 LEN Hostname 18d7fd04 180a + // 080a 12 1d08e807 12 180801 12 LEN Hostname 18d7fd04 180a + // 080a 12 2208e807 12 1d08e807 12 LEN Hostname 18d7fd04 180a + // Each chunk ends in 12 + // 4th chunk represent length of the name. + // 5th chunk is the name + int chunk = 0; + int i = 0; + String st = ""; + StringBuilder hostname = new StringBuilder(); + while (chunk < 3) { + st = "" + charArray[i] + "" + charArray[i + 1]; + if (DELIMITER_12.equals(st)) { + chunk++; + } + i += 2; + } + st = "" + charArray[i] + "" + charArray[i + 1]; + i += 2; + int length = Integer.parseInt(st, 16) * 2; + int current = i; + for (; i < current + length; i = i + 2) { + st = "" + charArray[i] + "" + charArray[i + 1]; + hostname.append(st); + } + logger.trace("{} - Shield Hostname: {} {}", thingId, hostname, length); + String encHostname = ShieldTVRequest.encodeMessage(hostname.toString()); + logger.debug("{} - Shield Hostname Encoded: {}", thingId, encHostname); + callback.setHostName(encHostname); + } else if (msg.startsWith(MESSAGE_HOSTNAME)) { + // Longer hostname reply + // 080b 12 5b08b510 12 TOTALLEN? 0a LEN Hostname 12 LEN IPADDR Padding? 22 LEN DeviceID 2a LEN arm64-v8a + // 2a LEN armeabi-v7a 2a LEN armeabi 180b + // It's possible for there to be more or less of the arm lists + logger.trace("{} - Longer Hostname Reply", thingId); + + int i = 20; + int length; + int current; + + // Hostname + String st = "" + charArray[i] + "" + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + + StringBuilder hostname = new StringBuilder(); + current = i; + + for (; i < current + length; i = i + 2) { + st = "" + charArray[i] + "" + charArray[i + 1]; + hostname.append(st); + } + + i += 2; // 12 + + // ipAddress + st = "" + charArray[i] + "" + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + + StringBuilder ipAddress = new StringBuilder(); + current = i; + + for (; i < current + length; i = i + 2) { + st = "" + charArray[i] + "" + charArray[i + 1]; + ipAddress.append(st); + } + + st = "" + charArray[i] + "" + charArray[i + 1]; + while (!DELIMITER_22.equals(st)) { + i += 2; + st = "" + charArray[i] + "" + charArray[i + 1]; + } + + i += 2; // 22 + + // deviceId + + st = "" + charArray[i] + "" + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + + StringBuilder deviceId = new StringBuilder(); + current = i; + + for (; i < current + length; i = i + 2) { + st = "" + charArray[i] + "" + charArray[i + 1]; + deviceId.append(st); + } + + // architectures + st = "" + charArray[i] + "" + charArray[i + 1]; + StringBuilder arch = new StringBuilder(); + while (DELIMITER_2A.equals(st)) { + i += 2; + st = "" + charArray[i] + "" + charArray[i + 1]; + length = Integer.parseInt(st, 16) * 2; + i += 2; + current = i; + for (; i < current + length; i = i + 2) { + st = "" + charArray[i] + "" + charArray[i + 1]; + arch.append(st); + } + st = "" + charArray[i] + "" + charArray[i + 1]; + if (DELIMITER_2A.equals(st)) { + arch.append("2c"); + } + } + + String encHostname = ShieldTVRequest.encodeMessage(hostname.toString()); + String encIpAddress = ShieldTVRequest.encodeMessage(ipAddress.toString()); + String encDeviceId = ShieldTVRequest.encodeMessage(deviceId.toString()); + String encArch = ShieldTVRequest.encodeMessage(arch.toString()); + logger.debug("{} - Hostname: {} - ipAddress: {} - deviceId: {} - arch: {}", thingId, encHostname, + encIpAddress, encDeviceId, encArch); + callback.setHostName(encHostname); + callback.setDeviceID(encDeviceId); + callback.setArch(encArch); + } else if (APP_START_SUCCESS.equals(msg)) { + // App successfully started + logger.debug("{} - App started successfully", thingId); + } else if (APP_START_FAILED.equals(msg)) { + // App failed to start + logger.debug("{} - App failed to start", thingId); + } else if (msg.startsWith(MESSAGE_APPDB) && msg.startsWith(DELIMITER_0A, 18)) { + // Individual update? + // 08f10712 5808061254 0a LEN app.name 12 LEN app.real.name 22 LEN URL 2801 300118f107 + logger.info("{} - Individual App Update - Please Report This: {}", thingId, msg); + } else if (msg.startsWith(MESSAGE_APPDB) && (msg.length() > 30)) { + // Massive dump of currently installed apps + // 08f10712 d81f080112 d31f0a540a LEN app.name 12 LEN app.real.name 22 LEN URL 2801 30010a650a LEN + Map appNameDB = new HashMap<>(); + Map appURLDB = new HashMap<>(); + int appCount = 0; + int i = 18; + String st = ""; + int length; + int current; + StringBuilder appSBPrepend = new StringBuilder(); + StringBuilder appSBDN = new StringBuilder(); + + // Load default apps that don't get sent in payload + + appNameDB.put("com.google.android.tvlauncher", "Android TV Home"); + appURLDB.put("com.google.android.tvlauncher", ""); + + appNameDB.put("com.google.android.katniss", "Google app for Android TV"); + appURLDB.put("com.google.android.katniss", ""); + + appNameDB.put("com.google.android.katnisspx", "Google app for Android TV (Pictures)"); + appURLDB.put("com.google.android.katnisspx", ""); + + appNameDB.put("com.google.android.backdrop", "Backdrop Daydream"); + appURLDB.put("com.google.android.backdrop", ""); + + // Packet will end with 300118f107 after last entry + + while (i < msg.length() - 10) { + StringBuilder appSBName = new StringBuilder(); + StringBuilder appSBURL = new StringBuilder(); + + // There are instances such as plex where multiple apps are sent as part of the same payload + // This is identified when 12 is the beginning of the set + + st = "" + charArray[i] + "" + charArray[i + 1]; + + if (!DELIMITER_12.equals(st)) { + appSBPrepend = new StringBuilder(); + appSBDN = new StringBuilder(); + + appCount++; + + // App Prepend + // Usually 10 in length but can be longer or shorter so look for 0a twice + do { + st = "" + charArray[i] + "" + charArray[i + 1]; + appSBPrepend.append(st); + i += 2; + } while (!DELIMITER_0A.equals(st)); + do { + st = "" + charArray[i] + "" + charArray[i + 1]; + appSBPrepend.append(st); + i += 2; + } while (!DELIMITER_0A.equals(st)); + st = "" + charArray[i] + "" + charArray[i + 1]; + + // Look for a third 0a, but only if 12 is not down the line + // If 12 is exactly 20 away from 0a that means that the DN was actually 10 long + String st2 = "" + charArray[i + 22] + "" + charArray[i + 23]; + if (DELIMITER_0A.equals(st.toString()) && !DELIMITER_12.equals(st2)) { + appSBPrepend.append(st); + i += 2; + st = "" + charArray[i] + "" + charArray[i + 1]; + } + + // app DN + length = Integer.parseInt(st, 16) * 2; + i += 2; + current = i; + for (; i < current + length; i = i + 2) { + st = "" + charArray[i] + "" + charArray[i + 1]; + appSBDN.append(st); + } + } else { + logger.trace("Second Entry"); + } + + // App Name + + i += 2; // 12 delimiter + st = "" + charArray[i] + "" + charArray[i + 1]; + i += 2; + length = Integer.parseInt(st, 16) * 2; + current = i; + for (; i < current + length; i = i + 2) { + st = "" + charArray[i] + "" + charArray[i + 1]; + appSBName.append(st); + } + + // There are times where there is padding here for no reason beyond the specified length. + // Proceed forward until we get to the 22 delimiter + + st = "" + charArray[i] + "" + charArray[i + 1]; + while (!DELIMITER_22.equals(st)) { + i += 2; + st = "" + charArray[i] + "" + charArray[i + 1]; + } + + // App URL + i += 2; // 22 delimiter + st = "" + charArray[i] + "" + charArray[i + 1]; + i += 2; + length = Integer.parseInt(st, 16) * 2; + current = i; + for (; i < current + length; i = i + 2) { + st = "" + charArray[i] + "" + charArray[i + 1]; + appSBURL.append(st); + } + st = "" + charArray[i] + "" + charArray[i + 1]; + if (!DELIMITER_12.equals(st)) { + i += 4; // terminates 2801 + } + String appPrepend = appSBPrepend.toString(); + String appDN = ShieldTVRequest.encodeMessage(appSBDN.toString()); + String appName = ShieldTVRequest.encodeMessage(appSBName.toString()); + String appURL = ShieldTVRequest.encodeMessage(appSBURL.toString()); + logger.debug("{} - AppPrepend: {} AppDN: {} AppName: {} AppURL: {}", thingId, appPrepend, appDN, + appName, appURL); + appNameDB.put(appDN, appName); + appURLDB.put(appDN, appURL); + } + if (appCount > 0) { + Map sortedAppNameDB = new LinkedHashMap<>(); + List valueList = new ArrayList<>(); + for (Map.Entry entry : appNameDB.entrySet()) { + valueList.add(entry.getValue()); + } + Collections.sort(valueList); + for (String str : valueList) { + for (Entry entry : appNameDB.entrySet()) { + if (entry.getValue().equals(str)) { + sortedAppNameDB.put(entry.getKey(), str); + } + } + } + + logger.trace("{} - MP appNameDB: {} sortedAppNameDB: {} appURLDB: {}", thingId, + appNameDB.toString(), sortedAppNameDB.toString(), appURLDB.toString()); + callback.setAppDB(sortedAppNameDB, appURLDB); + } else { + logger.warn("{} - MP empty msg: {} appDB appNameDB: {} appURLDB: {}", thingId, msg, + appNameDB.toString(), appURLDB.toString()); + } + } else if (msg.startsWith(MESSAGE_GOOD_COMMAND)) { + // This has something to do with successful command response, maybe. + } else if (KEEPALIVE_REPLY.equals(msg)) { + // Keepalive Reply + } else if (msg.startsWith(MESSAGE_LOWPRIV) && msg.startsWith(MESSAGE_PINSTART, 6)) { + // 080a 12 0308cf08 180a + logger.debug("PIN Process Started"); + } else if (msg.startsWith(MESSAGE_CERT_COMING) && msg.length() == 6) { + // This seems to be 20**** when observed. It is unclear what this does. + // This seems to send immediately before the certificate reply and as a reply to the pin being sent + } else if (msg.startsWith(MESSAGE_SUCCESS)) { + // Successful command received + // 08f007 12 0c 0804 12 08 0a0608 01100c200f 18f007 - GOOD LOGIN + // 08f007 12 LEN 0804 12 LEN 0a0608 01100c200f 18f007 + // + // 08f00712 0c 0804 12 08 0a0608 01100e200f 18f007 KEY_VOLDOWN + // 08f00712 0c 0804 12 08 0a0608 01100f200f 18f007 KEY_VOLUP + // 08f00712 0c 0804 12 08 0a0608 01200f2801 18f007 KEY_MUTE + logger.info("{} - Login Successful to {}", thingId, callback.getHostName()); + callback.setLoggedIn(true); + } else if (TIMEOUT.equals(msg)) { + // Timeout + // 080a 12 1108b510 12 0c0804 12 08 54696d65206f7574 180a + // 080a 12 1108b510 12 0c0804 12 LEN Timeout 180a + logger.debug("{} - Timeout {}", thingId, msg); + } else if (msg.startsWith(MESSAGE_APP_SUCCESS) && msg.startsWith(MESSAGE_APP_GET_SUCCESS, 10)) { + // Get current app command successful. Usually paired with 0807 reply below. + } else if (msg.startsWith(MESSAGE_APP_SUCCESS) && msg.startsWith(MESSAGE_APP_CURRENT, 10)) { + // Current App + // 08ec07 12 2a0807 22 262205 656e5f555342 1d 636f6d2e676f6f676c652e616e64726f69642e74766c61756e63686572 + // 18ec07 + // 08ec07 12 2a0807 22 262205 en_USB LEN AppName 18ec07 + StringBuilder appName = new StringBuilder(); + String lengthStr = "" + charArray[34] + charArray[35]; + int length = Integer.parseInt(lengthStr, 16) * 2; + for (int i = 36; i < 36 + length; i++) { + appName.append(charArray[i]); + } + logger.debug("{} - Current App: {}", thingId, ShieldTVRequest.encodeMessage(appName.toString())); + callback.setCurrentApp(ShieldTVRequest.encodeMessage(appName.toString())); + } else if (msg.startsWith(MESSAGE_LOWPRIV) && msg.startsWith(MESSAGE_CERT, 10)) { + // Certificate Reply + // |--6-----------12-----------10---------------16---------6--- = 50 characters long + // |080a 12 ad10 08b510 12 a710 0801 12 07 53756363657373 1ac009 3082... 3082... 180a + // |080a 12 9f10 08b510 12 9910 0801 12 07 53756363657373 1ac209 3082... 3082... 180a + // |--------Little Endian Total Payload Length + // |-----------------------Little Endian Remaining Payload Length + // |-----------------------------------Length of SUCCESS + // |--------------------------------------ASCII: SUCCESS + // |-----------------------------------------------------Little Endian Length (e.g. 09c0 and 09c2 above) + // |------------------------------------------------------------Priv Key RSA 2048 + // |--------------------------------------------------------------------Cert X.509 + if (msg.startsWith(MESSAGE_CERT_PAYLOAD, 28)) { + StringBuilder preamble = new StringBuilder(); + StringBuilder privKey = new StringBuilder(); + StringBuilder pubKey = new StringBuilder(); + int i = 0; + int current; + for (; i < 44; i++) { + preamble.append(charArray[i]); + } + logger.trace("{} - Cert Preamble: {}", thingId, preamble.toString()); + + i += 2; // 1a + String st = "" + charArray[i + 2] + "" + charArray[i + 3] + "" + charArray[i] + "" + + charArray[i + 1]; + int privLen = 2246 + ((Integer.parseInt(st, 16) - 2400) * 2); + i += 4; // length + current = i; + + logger.trace("{} - Cert privLen: {} {}", thingId, st, privLen); + + for (; i < current + privLen; i++) { + privKey.append(charArray[i]); + } + + logger.trace("{} - Cert privKey: {} {}", thingId, privLen, privKey.toString()); + + for (; i < msg.length() - 4; i++) { + pubKey.append(charArray[i]); + } + + logger.trace("{} - Cert pubKey: {} {}", thingId, msg.length() - privLen - 4, pubKey.toString()); + + logger.debug("{} - Cert Pair Received privLen: {} pubLen: {}", thingId, privLen, + msg.length() - privLen - 4); + + byte[] privKeyB64Byte = DatatypeConverter.parseHexBinary(privKey.toString()); + byte[] pubKeyB64Byte = DatatypeConverter.parseHexBinary(pubKey.toString()); + + String privKeyB64 = Base64.getEncoder().encodeToString(privKeyB64Byte); + String pubKeyB64 = Base64.getEncoder().encodeToString(pubKeyB64Byte); + + callback.setKeys(privKeyB64, pubKeyB64); + } else { + logger.info("{} - Pin Process Failed.", thingId); + } + } else { + logger.info("{} - Unknown payload received. {}", thingId, msg); + } + } catch (Exception e) { + logger.info("{} - Message Parser Caught Exception", thingId, e); + } + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVRequest.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVRequest.java new file mode 100644 index 000000000..051ac5407 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/protocol/shieldtv/ShieldTVRequest.java @@ -0,0 +1,103 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.protocol.shieldtv; + +import static org.openhab.binding.androidtv.internal.AndroidTVBindingConstants.*; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * Contains static methods for constructing LEAP messages + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class ShieldTVRequest { + + public static String encodeMessage(String message) { + StringBuilder reply = new StringBuilder(); + char[] charArray = message.toCharArray(); + for (int i = 0; i < charArray.length; i = i + 2) { + String st = "" + charArray[i] + "" + charArray[i + 1]; + char ch = (char) Integer.parseInt(st, 16); + reply.append(ch); + } + return reply.toString(); + } + + public static String decodeMessage(String message) { + StringBuilder sb = new StringBuilder(); + char ch[] = message.toCharArray(); + for (int i = 0; i < ch.length; i++) { + String hexString = Integer.toHexString(ch[i]); + if (hexString.length() % 2 > 0) { + sb.append('0'); + } + sb.append(hexString); + } + return sb.toString(); + } + + public static String pinRequest(String pin) { + if (PIN_REQUEST.equals(pin)) { + String message = "080a120308cd08"; + return message; + } else { + String prefix = "080a121f08d108121a0a06"; + String encodedPin = decodeMessage(pin); + String suffix = "121036646564646461326639366635646261"; + return prefix + encodedPin + suffix; + } + } + + public static String loginRequest() { + String message = "0801121a0801121073616d73756e6720534d2d4739393855180128fbff04"; + return message; + } + + public static String keepAlive() { + String message = "080028fae0a6c0d130"; + return message; + } + + private static String fixMessage(String tempMsg) { + if (tempMsg.length() % 2 > 0) { + tempMsg = "0" + tempMsg; + } + return tempMsg; + } + + public static String startApp(String message) { + int length = message.length(); + String len1 = fixMessage(Integer.toHexString(length + 6)); + String len2 = fixMessage(Integer.toHexString(length + 2)); + String len3 = fixMessage(Integer.toHexString(length)); + String reply = "08f10712" + len1 + "080212" + len2 + "0a" + len3 + decodeMessage(message); + return reply; + } + // 080b120308cd08 - Longer Hostname Reply + // 08f30712020805 - Unknown + // 08f10712020800 - Get all apps + // 08ec0712020806 - Get current app + + public static String keyboardEntry(String entry) { + // 08ec07120d08081205616263646532020a0a + // 08ec0712 0d 0808 12 05 6162636465 3202 0a0a + int length = entry.length(); + String len1 = fixMessage(Integer.toHexString(length + 8)); + String len2 = fixMessage(Integer.toHexString(length)); + String len3 = fixMessage(Integer.toHexString(length * 2)); + String reply = "08ec0712" + len1 + "080812" + len2 + decodeMessage(entry) + "3202" + len3 + len3; + return reply; + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/utils/AndroidTVPKI.java b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/utils/AndroidTVPKI.java new file mode 100644 index 000000000..aad3a8a89 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/java/org/openhab/binding/androidtv/internal/utils/AndroidTVPKI.java @@ -0,0 +1,291 @@ +/** + * Copyright (c) 2010-2023 Contributors to the openHAB project + * + * See the NOTICE file(s) distributed with this work for additional + * information. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0 + * + * SPDX-License-Identifier: EPL-2.0 + */ +package org.openhab.binding.androidtv.internal.utils; + +import java.io.ByteArrayInputStream; +import java.io.FileInputStream; +import java.io.FileOutputStream; +import java.io.IOException; +import java.math.BigInteger; +import java.nio.charset.StandardCharsets; +import java.security.GeneralSecurityException; +import java.security.Key; +import java.security.KeyFactory; +import java.security.KeyPair; +import java.security.KeyPairGenerator; +import java.security.KeyStore; +import java.security.NoSuchAlgorithmException; +import java.security.Security; +import java.security.Signature; +import java.security.cert.Certificate; +import java.security.cert.CertificateEncodingException; +import java.security.cert.CertificateException; +import java.security.cert.CertificateFactory; +import java.security.cert.X509Certificate; +import java.security.spec.PKCS8EncodedKeySpec; +import java.time.Duration; +import java.time.Instant; +import java.util.Base64; +import java.util.Date; + +import javax.crypto.Cipher; +import javax.crypto.KeyGenerator; +import javax.crypto.NoSuchPaddingException; +import javax.crypto.spec.GCMParameterSpec; +import javax.crypto.spec.SecretKeySpec; + +import org.bouncycastle.asn1.x500.X500Name; +import org.bouncycastle.cert.X509v3CertificateBuilder; +import org.bouncycastle.cert.jcajce.JcaX509CertificateConverter; +import org.bouncycastle.cert.jcajce.JcaX509v3CertificateBuilder; +import org.bouncycastle.jce.provider.BouncyCastleProvider; +import org.bouncycastle.operator.ContentSigner; +import org.bouncycastle.operator.OperatorCreationException; +import org.bouncycastle.operator.jcajce.JcaContentSignerBuilder; +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +/** + * The {@link AndroidTVPKI} class controls all aspects of the PKI/keyStore + * + * Some methods adapted from Bosch binding + * + * @author Ben Rosenblum - Initial contribution + */ +@NonNullByDefault +public class AndroidTVPKI { + + private final Logger logger = LoggerFactory.getLogger(AndroidTVPKI.class); + + private final int keySize = 128; + private final int dataLength = 128; + + private String privKey = ""; + private String cert = ""; + private String keystoreFileName = ""; + private String keystoreAlgorithm = "RSA"; + private int keyLength = 2048; + private String alias = "openhab"; + private String distName = "CN=openHAB, O=openHAB, L=None, ST=None, C=None"; + private String cipher = "AES/GCM/NoPadding"; + private String keyAlgorithm = ""; + + private @Nullable Cipher encryptionCipher; + + public AndroidTVPKI() { + try { + encryptionCipher = Cipher.getInstance(cipher); + } catch (NoSuchAlgorithmException | NoSuchPaddingException e) { + logger.debug("Could not get cipher instance", e); + } + } + + public byte[] generateEncryptionKey() { + Key key; + try { + KeyGenerator keyGenerator = KeyGenerator.getInstance("AES"); + keyGenerator.init(keySize); + key = keyGenerator.generateKey(); + byte[] newKey = key.getEncoded(); + this.keyAlgorithm = key.getAlgorithm(); + return newKey; + } catch (NoSuchAlgorithmException e) { + logger.debug("Could not generate encryption keys", e); + } + return new byte[0]; + } + + private Key convertByteToKey(byte[] keyString) { + Key key = new SecretKeySpec(keyString, keyAlgorithm); + return key; + } + + public String encrypt(String data, Key key) throws Exception { + return encrypt(data, key, this.cipher); + } + + public String encrypt(String data, Key key, String cipher) throws Exception { + byte[] dataInBytes = data.getBytes(); + Cipher encryptionCipher = this.encryptionCipher; + if (encryptionCipher != null) { + encryptionCipher.init(Cipher.ENCRYPT_MODE, key); + byte[] encryptedBytes = encryptionCipher.doFinal(dataInBytes); + return Base64.getEncoder().encodeToString(encryptedBytes); + } else { + return ""; + } + } + + public String decrypt(String encryptedData, Key key) throws Exception { + return decrypt(encryptedData, key, this.cipher); + } + + public String decrypt(String encryptedData, Key key, String cipher) throws Exception { + byte[] dataInBytes = Base64.getDecoder().decode(encryptedData); + Cipher decryptionCipher = Cipher.getInstance(cipher); + Cipher encryptionCipher = this.encryptionCipher; + if (encryptionCipher != null) { + GCMParameterSpec spec = new GCMParameterSpec(dataLength, encryptionCipher.getIV()); + decryptionCipher.init(Cipher.DECRYPT_MODE, key, spec); + byte[] decryptedBytes = decryptionCipher.doFinal(dataInBytes); + return new String(decryptedBytes); + } else { + return ""; + } + } + + public void setPrivKey(String privKey, byte[] keyString) throws Exception { + Key key = convertByteToKey(keyString); + this.privKey = encrypt(privKey, key); + } + + public String getPrivKey(byte[] keyString) throws Exception { + Key key = convertByteToKey(keyString); + return decrypt(this.privKey, key); + } + + public void setCert(String cert) { + this.cert = cert; + } + + public void setCert(Certificate cert) throws CertificateEncodingException { + this.cert = new String(Base64.getEncoder().encode(cert.getEncoded())); + } + + public Certificate getCert() throws CertificateException { + Certificate cert = CertificateFactory.getInstance("X.509") + .generateCertificate(new ByteArrayInputStream(Base64.getDecoder().decode(this.cert.getBytes()))); + return cert; + } + + public void setAlias(String alias) { + this.alias = alias; + } + + public String getAlias() { + return this.alias; + } + + public void setAlgorithm(String keystoreAlgorithm) { + this.keystoreAlgorithm = keystoreAlgorithm; + } + + public String getAlgorithm() { + return this.keystoreAlgorithm; + } + + public void setKeyLength(int keyLength) { + this.keyLength = keyLength; + } + + public int getKeyLength() { + return this.keyLength; + } + + public void setDistName(String distName) { + this.distName = distName; + } + + public String getDistName() { + return this.distName; + } + + public void setKeystoreFileName(String keystoreFileName) { + this.keystoreFileName = keystoreFileName; + } + + public String getKeystoreFileName() { + return this.keystoreFileName; + } + + public void setKeys(String privKey, byte[] keyString, String cert) throws GeneralSecurityException, Exception { + setPrivKey(privKey, keyString); + setCert(cert); + } + + public void setKeyStore(String keystoreFileName) { + this.keystoreFileName = keystoreFileName; + } + + public void loadFromKeyStore(String keystoreFileName, String keystorePassword, byte[] keyString) + throws GeneralSecurityException, IOException, Exception { + this.keystoreFileName = keystoreFileName; + loadFromKeyStore(keystorePassword, keyString); + } + + public void loadFromKeyStore(String keystorePassword, byte[] keyString) + throws GeneralSecurityException, IOException, Exception { + Key key = convertByteToKey(keyString); + KeyStore keystore = KeyStore.getInstance("JKS"); + FileInputStream keystoreInputStream = new FileInputStream(this.keystoreFileName); + keystore.load(keystoreInputStream, keystorePassword.toCharArray()); + byte[] byteKey = keystore.getKey(this.alias, keystorePassword.toCharArray()).getEncoded(); + this.privKey = encrypt(new String(Base64.getEncoder().encode(byteKey)), key); + setCert(keystore.getCertificate(this.alias)); + } + + public KeyStore getKeyStore(String keystorePassword, byte[] keyString) + throws GeneralSecurityException, IOException, Exception { + KeyStore keystore = KeyStore.getInstance("JKS"); + keystore.load(null, null); + byte[] pkcs8EncodedBytes = Base64.getDecoder().decode(getPrivKey(keyString)); + PKCS8EncodedKeySpec keySpec = new PKCS8EncodedKeySpec(pkcs8EncodedBytes); + KeyFactory kf = KeyFactory.getInstance(this.keystoreAlgorithm); + keystore.setKeyEntry(this.alias, kf.generatePrivate(keySpec), keystorePassword.toCharArray(), + new java.security.cert.Certificate[] { getCert() }); + return keystore; + } + + public void saveKeyStore(String keystorePassword, byte[] keyString) + throws GeneralSecurityException, IOException, Exception { + saveKeyStore(this.keystoreFileName, keystorePassword, keyString); + } + + public void saveKeyStore(String keystoreFileName, String keystorePassword, byte[] keyString) + throws GeneralSecurityException, IOException, Exception { + FileOutputStream keystoreStream = new FileOutputStream(keystoreFileName); + KeyStore keystore = getKeyStore(keystorePassword, keyString); + keystore.store(keystoreStream, keystorePassword.toCharArray()); + } + + private X509Certificate generateSelfSignedCertificate(KeyPair keyPair, String distName) + throws GeneralSecurityException, OperatorCreationException { + final Instant now = Instant.now(); + final Date notBefore = Date.from(now); + final Date notAfter = Date.from(now.plus(Duration.ofDays(365 * 10))); + X500Name name = new X500Name(distName); + X509v3CertificateBuilder certificateBuilder = new JcaX509v3CertificateBuilder(name, + BigInteger.valueOf(now.toEpochMilli()), notBefore, notAfter, name, keyPair.getPublic()); + ContentSigner contentSigner = new JcaContentSignerBuilder("SHA256WithRSA").build(keyPair.getPrivate()); + return new JcaX509CertificateConverter().setProvider(new BouncyCastleProvider()) + .getCertificate(certificateBuilder.build(contentSigner)); + } + + public void generateNewKeyPair(byte[] keyString) + throws GeneralSecurityException, OperatorCreationException, IOException, Exception { + Key key = convertByteToKey(keyString); + KeyPairGenerator kpg = KeyPairGenerator.getInstance(this.keystoreAlgorithm); + kpg.initialize(this.keyLength); + KeyPair kp = kpg.generateKeyPair(); + Security.addProvider(new BouncyCastleProvider()); + Signature signer = Signature.getInstance("SHA256withRSA", "BC"); + signer.initSign(kp.getPrivate()); + signer.update("openhab".getBytes(StandardCharsets.UTF_8)); + signer.sign(); + X509Certificate signedcert = generateSelfSignedCertificate(kp, this.distName); + this.privKey = encrypt(new String(Base64.getEncoder().encode(kp.getPrivate().getEncoded())), key); + setCert(signedcert); + } +} diff --git a/bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/addon/addon.xml b/bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/addon/addon.xml new file mode 100644 index 000000000..dde94687e --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/addon/addon.xml @@ -0,0 +1,11 @@ + + + + binding + AndroidTV Binding + This is the add-on for AndroidTV. + local + + diff --git a/bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/i18n/androidtv.properties b/bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/i18n/androidtv.properties new file mode 100644 index 000000000..d8a2799c2 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/i18n/androidtv.properties @@ -0,0 +1,69 @@ +# add-on + +addon.androidtv.name = AndroidTV Binding +addon.androidtv.description = This is the add-on for AndroidTV. + +# thing types + +thing-type.androidtv.googletv.label = GoogleTV +thing-type.androidtv.googletv.description = GoogleTV +thing-type.androidtv.shieldtv.label = ShieldTV +thing-type.androidtv.shieldtv.description = Nvidia ShieldTV + +# thing types config + +thing-type.config.androidtv.googletv.delay.label = Delay +thing-type.config.androidtv.googletv.delay.description = Delay between messages +thing-type.config.androidtv.googletv.heartbeat.label = Heartbeat Frequency +thing-type.config.androidtv.googletv.heartbeat.description = Frequency of heartbeats +thing-type.config.androidtv.googletv.ipAddress.label = Hostname +thing-type.config.androidtv.googletv.ipAddress.description = Hostname or IP address of the device +thing-type.config.androidtv.googletv.keystoreFileName.label = Keystore File Name +thing-type.config.androidtv.googletv.keystoreFileName.description = Java keystore containing key and certs +thing-type.config.androidtv.googletv.keystorePassword.label = Keystore Password +thing-type.config.androidtv.googletv.keystorePassword.description = Password for the keystore file +thing-type.config.androidtv.googletv.port.label = Port +thing-type.config.androidtv.googletv.port.description = Port to connect to +thing-type.config.androidtv.googletv.reconnect.label = Reconnect Delay +thing-type.config.androidtv.googletv.reconnect.description = Delay between reconnection attempts +thing-type.config.androidtv.shieldtv.delay.label = Delay +thing-type.config.androidtv.shieldtv.delay.description = Delay between messages +thing-type.config.androidtv.shieldtv.heartbeat.label = Hearbeat Frequency +thing-type.config.androidtv.shieldtv.heartbeat.description = Frequency of heartbeats +thing-type.config.androidtv.shieldtv.ipAddress.label = Hostname +thing-type.config.androidtv.shieldtv.ipAddress.description = Hostname or IP address of the device +thing-type.config.androidtv.shieldtv.keystoreFileName.label = Keystore File Name +thing-type.config.androidtv.shieldtv.keystoreFileName.description = Java keystore containing key and certs +thing-type.config.androidtv.shieldtv.keystorePassword.label = Keystore Password +thing-type.config.androidtv.shieldtv.keystorePassword.description = Password for the keystore file +thing-type.config.androidtv.shieldtv.port.label = Port +thing-type.config.androidtv.shieldtv.port.description = Port to connect to +thing-type.config.androidtv.shieldtv.reconnect.label = Reconnect Delay +thing-type.config.androidtv.shieldtv.reconnect.description = Delay between reconnection attempts + +# channel types + +channel-type.androidtv.app.label = App +channel-type.androidtv.app.description = App Control +channel-type.androidtv.appname.label = App Name +channel-type.androidtv.appname.description = App Name +channel-type.androidtv.appurl.label = App URL +channel-type.androidtv.appurl.description = App URL +channel-type.androidtv.debug.label = DEBUG Command +channel-type.androidtv.debug.description = Binding control (for debugging) +channel-type.androidtv.keyboard.label = Keyboard +channel-type.androidtv.keyboard.description = Keyboard Entry +channel-type.androidtv.keycode.label = Keycode +channel-type.androidtv.keycode.description = Send keycode +channel-type.androidtv.keypress.label = Key Press +channel-type.androidtv.keypress.description = Send key press +channel-type.androidtv.pincode.label = Pin Code +channel-type.androidtv.pincode.description = Send Pin Code +channel-type.androidtv.player.label = Player +channel-type.androidtv.player.description = Player Control + +# custom thing status +offline.protocols-starting = Protocols Starting +offline.googletv-address-not-specified = googletv address not specified +offline.shieldtv-address-not-specified = shieldtv address not specified + diff --git a/bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/thing/thing-types.xml b/bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/thing/thing-types.xml new file mode 100644 index 000000000..9101a2504 --- /dev/null +++ b/bundles/org.openhab.binding.androidtv/src/main/resources/OH-INF/thing/thing-types.xml @@ -0,0 +1,192 @@ + + + + + + + Nvidia ShieldTV + + + + + + + + + + + + + + + + + + unknown + unknown + unknown + unknown + unknown + unknown + unknown + unknown + + + ipAddress + + + + network-address + + Hostname or IP address of the device + + + + Port to connect to + + + + Java keystore containing key and certs + + + password + + Password for the keystore file + + + + Delay between reconnection attempts + + + + Frequency of heartbeats + + + + Delay between messages + + + + + + + + + GoogleTV + + + + + + + + + + + + + + + + unknown + unknown + unknown + unknown + unknown + + + ipAddress + + + + network-address + + Hostname or IP address of the device + + + + Port to connect to + + + + Java keystore containing key and certs + + + password + + Password for the keystore file + + + + Delay between reconnection attempts + + + + Frequency of heartbeats + + + + Delay between messages + + + + + + + String + + Binding control (for debugging) + + + + String + + App Control + + + + String + + App Name + + + + String + + App URL + + + + String + + Send key press + + + + String + + Send keycode + + + + String + + Keyboard Entry + + + + String + + Send Pin Code + + + + Player + + Player Control + + + diff --git a/bundles/pom.xml b/bundles/pom.xml index 5fb1ccee8..cf42108c5 100644 --- a/bundles/pom.xml +++ b/bundles/pom.xml @@ -56,6 +56,7 @@ org.openhab.binding.ambientweather org.openhab.binding.amplipi org.openhab.binding.androiddebugbridge + org.openhab.binding.androidtv org.openhab.binding.anel org.openhab.binding.anthem org.openhab.binding.astro