/* * Copyright 2015 Synced Synapse. All rights reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.xbmc.kore.ui.sections.hosts; import android.app.Activity; import android.app.ProgressDialog; import android.content.DialogInterface; import android.os.Bundle; import android.os.Handler; import android.support.v4.app.Fragment; import android.text.TextUtils; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.Button; import android.widget.CheckBox; import android.widget.CompoundButton; import android.widget.EditText; import android.widget.Toast; import org.xbmc.kore.R; import org.xbmc.kore.eventclient.EventServerConnection; import org.xbmc.kore.host.HostInfo; import org.xbmc.kore.jsonrpc.ApiCallback; import org.xbmc.kore.jsonrpc.ApiException; import org.xbmc.kore.jsonrpc.HostConnection; import org.xbmc.kore.jsonrpc.method.Application; import org.xbmc.kore.jsonrpc.method.JSONRPC; import org.xbmc.kore.jsonrpc.type.ApplicationType; import org.xbmc.kore.utils.LogUtils; import org.xbmc.kore.utils.NetUtils; import java.util.regex.Matcher; import java.util.regex.Pattern; import butterknife.ButterKnife; import butterknife.BindView; import butterknife.Unbinder; /** * Fragment that presents the welcome message */ public class HostFragmentManualConfiguration extends Fragment { private static final String TAG = LogUtils.makeLogTag(HostFragmentManualConfiguration.class); /** * Fragment arguments */ private static final String PREFIX = "org.xbmc.kore"; public static final String HOST_ID = PREFIX + ".host_id", HOST_NAME = PREFIX + ".host_name", HOST_ADDRESS = PREFIX + ".host_address", HOST_HTTP_PORT = PREFIX + ".host_http_port", HOST_TCP_PORT = PREFIX + ".host_tcp_post", HOST_USERNAME = PREFIX + ".host_username", HOST_PASSWORD = PREFIX + ".host_password", HOST_MAC_ADDRESS = PREFIX + ".host_mac_address", HOST_WOL_PORT = PREFIX + ".host_wol_port", HOST_PROTOCOL = PREFIX + ".host_protocol", HOST_USE_EVENT_SERVER = PREFIX + ".host_use_event_server", HOST_EVENT_SERVER_PORT = PREFIX + ".host_event_server_port"; public static final String GO_STRAIGHT_TO_TEST = PREFIX + ".go_straight_to_test"; /** * Callback interface to communicate with the encolsing activity */ public interface HostManualConfigurationListener { public void onHostManualConfigurationNext(HostInfo hostInfo); public void onHostManualConfigurationCancel(); } public static String CANCEL_BUTTON_LABEL_ARG = PREFIX + ".cancel_button_label"; private HostManualConfigurationListener listener; private ProgressDialog progressDialog; private Unbinder unbinder; @BindView(R.id.xbmc_name) EditText xbmcNameEditText; @BindView(R.id.xbmc_ip_address) EditText xbmcIpAddressEditText; @BindView(R.id.xbmc_http_port) EditText xbmcHttpPortEditText; @BindView(R.id.xbmc_tcp_port) EditText xbmcTcpPortEditText; @BindView(R.id.xbmc_username) EditText xbmcUsernameEditText; @BindView(R.id.xbmc_password) EditText xbmcPasswordEditText; @BindView(R.id.xbmc_mac_address) EditText xbmcMacAddressEditText; @BindView(R.id.xbmc_wol_port) EditText xbmcWolPortEditText; @BindView(R.id.xbmc_use_tcp) CheckBox xbmcUseTcpCheckbox; @BindView(R.id.xbmc_use_event_server) CheckBox xbmcUseEventServerCheckbox; @BindView(R.id.xbmc_event_server_port) EditText xbmcEventServerPortEditText; // Handler for callbacks final Handler handler = new Handler(); @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View root = inflater.inflate(R.layout.fragment_add_host_manual_configuration, container, false); unbinder = ButterKnife.bind(this, root); // By default, use TCP xbmcUseTcpCheckbox.setChecked(true); xbmcUseTcpCheckbox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { @Override public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { xbmcTcpPortEditText.setEnabled(isChecked); } }); xbmcUseEventServerCheckbox.setChecked(true); xbmcUseEventServerCheckbox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { @Override public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { xbmcEventServerPortEditText.setEnabled(isChecked); } }); // Check if we were given a host info String hostName = getArguments().getString(HOST_NAME); String hostAddress = getArguments().getString(HOST_ADDRESS); int hostHttpPort = getArguments().getInt(HOST_HTTP_PORT, HostInfo.DEFAULT_HTTP_PORT); int hostTcpPort = getArguments().getInt(HOST_TCP_PORT, HostInfo.DEFAULT_TCP_PORT); String hostUsername = getArguments().getString(HOST_USERNAME); String hostPassword = getArguments().getString(HOST_PASSWORD); int hostProtocol = getArguments().getInt(HOST_PROTOCOL, HostConnection.PROTOCOL_TCP); String hostMacAddress = getArguments().getString(HOST_MAC_ADDRESS); int hostWolPort = getArguments().getInt(HOST_WOL_PORT, HostInfo.DEFAULT_WOL_PORT); boolean hostUseEventServer = getArguments().getBoolean(HOST_USE_EVENT_SERVER, true); int hostEventServerPort = getArguments().getInt(HOST_EVENT_SERVER_PORT, HostInfo.DEFAULT_EVENT_SERVER_PORT); if (hostAddress != null) { xbmcNameEditText.setText(hostName); xbmcIpAddressEditText.setText(hostAddress); xbmcHttpPortEditText.setText(String.valueOf(hostHttpPort)); if (!TextUtils.isEmpty(hostUsername)) xbmcUsernameEditText.setText(hostUsername); if (!TextUtils.isEmpty(hostPassword)) xbmcPasswordEditText.setText(hostPassword); xbmcUseTcpCheckbox.setChecked(!(hostProtocol == HostConnection.PROTOCOL_HTTP)); xbmcTcpPortEditText.setEnabled(xbmcUseTcpCheckbox.isChecked()); if (hostTcpPort != HostInfo.DEFAULT_TCP_PORT) xbmcTcpPortEditText.setText(String.valueOf(hostTcpPort)); if (!TextUtils.isEmpty(hostMacAddress)) xbmcMacAddressEditText.setText(hostMacAddress); if (hostWolPort != HostInfo.DEFAULT_WOL_PORT) xbmcWolPortEditText.setText(String.valueOf(hostWolPort)); xbmcUseEventServerCheckbox.setChecked(hostUseEventServer); xbmcEventServerPortEditText.setEnabled(xbmcUseEventServerCheckbox.isChecked()); if (hostEventServerPort != HostInfo.DEFAULT_EVENT_SERVER_PORT) xbmcEventServerPortEditText.setText(String.valueOf(hostEventServerPort)); } return root; } @Override public void onActivityCreated(Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); if (getView() == null) return; progressDialog = new ProgressDialog(getActivity()); Button next, previous; // Next button next = (Button)getView().findViewById(R.id.next); next.setText(R.string.test_connection); next.setCompoundDrawables(null, null, null, null); next.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { testConnection(); } }); // Previous button previous = (Button)getView().findViewById(R.id.previous); if (getArguments().getString(CANCEL_BUTTON_LABEL_ARG, null) != null) { previous.setText(getArguments().getString(CANCEL_BUTTON_LABEL_ARG)); } else { previous.setText(android.R.string.cancel); } previous.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { listener.onHostManualConfigurationCancel(); } }); // Check if the activity wants us to go straight to test boolean goStraighToTest = getArguments().getBoolean(GO_STRAIGHT_TO_TEST, false); if (goStraighToTest) { testConnection(); } } @Override public void onAttach(Activity activity) { super.onAttach(activity); try { listener = (HostManualConfigurationListener) activity; } catch (ClassCastException e) { throw new ClassCastException(activity.toString() + " must implement AddHostManualConfigurationListener interface."); } } @Override public void onDestroyView() { super.onDestroyView(); unbinder.unbind(); } private static boolean isValidPort(int port) { return port > 0 && port <= 65535; } /** * Tests a connection with the values set in the UI. * Checks whether the values are correctly set, and then tries to make * a ping call. First through HTTP, and if it succeeds, through TCP to * check availability. Finally adds the host and advances the wizard */ private void testConnection() { String xbmcName = xbmcNameEditText.getText().toString(); boolean isHttps = false; String xbmcAddress = xbmcIpAddressEditText.getText().toString(); if (xbmcAddress.startsWith("https://")) { xbmcAddress = xbmcAddress.substring("https://".length()); LogUtils.LOGD(TAG, "Stripped https:// on address to get: " + xbmcAddress); isHttps = true; } else if (xbmcAddress.startsWith("http://")) { xbmcAddress = xbmcAddress.substring("http://".length()); LogUtils.LOGD(TAG, "Stripped http:// on address to get: " + xbmcAddress); } int xbmcHttpPort = isHttps ? HostInfo.DEFAULT_HTTPS_PORT : HostInfo.DEFAULT_HTTP_PORT; Integer implicitPort = null; Matcher m = Pattern.compile("^.*:(\\d{1,5})\\z").matcher(xbmcAddress); if (m.matches()) { // Minus one character for the colon xbmcAddress = xbmcAddress.substring(0, m.start(1) - 1); LogUtils.LOGD(TAG, "Stripped port on address to get: " + xbmcAddress); try { implicitPort = Integer.valueOf(m.group(1)); } catch (NumberFormatException e) { LogUtils.LOGW( TAG, "Value matching port regex couldn't be parsed as integer: " + m.group(1) ); implicitPort = -1; } } Integer explicitPort = null; String aux = xbmcHttpPortEditText.getText().toString(); if (!TextUtils.isEmpty(aux)) { try { explicitPort = Integer.valueOf(aux); } catch (NumberFormatException e) { explicitPort = -1; } } if (implicitPort != null) { if (!isValidPort(implicitPort)) { Toast.makeText(getActivity(), R.string.wizard_invalid_http_port_specified, Toast.LENGTH_SHORT); xbmcIpAddressEditText.requestFocus(); return; } xbmcHttpPort = implicitPort; } else if (explicitPort != null) { if (!isValidPort(explicitPort)) { Toast.makeText(getActivity(), R.string.wizard_invalid_http_port_specified, Toast.LENGTH_SHORT); xbmcHttpPortEditText.requestFocus(); return; } xbmcHttpPort = explicitPort; } String xbmcUsername = xbmcUsernameEditText.getText().toString(); String xbmcPassword = xbmcPasswordEditText.getText().toString(); aux = xbmcTcpPortEditText.getText().toString(); int xbmcTcpPort; try { xbmcTcpPort = TextUtils.isEmpty(aux) ? HostInfo.DEFAULT_TCP_PORT : Integer.valueOf(aux); } catch (NumberFormatException exc) { xbmcTcpPort = -1; } int xbmcProtocol = xbmcUseTcpCheckbox.isChecked()? HostConnection.PROTOCOL_TCP : HostConnection.PROTOCOL_HTTP; String macAddress = xbmcMacAddressEditText.getText().toString(); aux = xbmcWolPortEditText.getText().toString(); int xbmcWolPort = HostInfo.DEFAULT_WOL_PORT; try { xbmcWolPort = TextUtils.isEmpty(aux) ? HostInfo.DEFAULT_WOL_PORT : Integer.valueOf(aux); } catch (NumberFormatException exc) { // Ignoring this exception and keeping WoL port at the default value } boolean xbmcUseEventServer = xbmcUseEventServerCheckbox.isChecked(); aux = xbmcEventServerPortEditText.getText().toString(); int xbmcEventServerPort; try { xbmcEventServerPort = TextUtils.isEmpty(aux) ? HostInfo.DEFAULT_EVENT_SERVER_PORT : Integer.valueOf(aux); } catch (NumberFormatException exc) { xbmcEventServerPort = -1; } // Check Xbmc name and address if (TextUtils.isEmpty(xbmcName)) { Toast.makeText(getActivity(), R.string.wizard_no_name_specified, Toast.LENGTH_SHORT).show(); xbmcNameEditText.requestFocus(); return; } else if (TextUtils.isEmpty(xbmcAddress)) { Toast.makeText(getActivity(), R.string.wizard_no_address_specified, Toast.LENGTH_SHORT).show(); xbmcIpAddressEditText.requestFocus(); return; } else if (xbmcTcpPort <= 0) { Toast.makeText(getActivity(), R.string.wizard_invalid_tcp_port_specified, Toast.LENGTH_SHORT).show(); xbmcTcpPortEditText.requestFocus(); return; } else if (xbmcEventServerPort <= 0) { Toast.makeText(getActivity(), R.string.wizard_invalid_tcp_port_specified, Toast.LENGTH_SHORT).show(); xbmcEventServerPortEditText.requestFocus(); return; } // If username or password empty, set it to null if (TextUtils.isEmpty(xbmcUsername)) xbmcUsername = null; if (TextUtils.isEmpty(xbmcPassword)) xbmcPassword = null; // Ok, let's try to ping the host final HostInfo checkedHostInfo = new HostInfo(xbmcName, xbmcAddress, xbmcProtocol, xbmcHttpPort, xbmcTcpPort, xbmcUsername, xbmcPassword, xbmcUseEventServer, xbmcEventServerPort, isHttps); checkedHostInfo.setMacAddress(macAddress); checkedHostInfo.setWolPort(xbmcWolPort); progressDialog.setTitle(String.format(getResources().getString(R.string.wizard_connecting_to_xbmc_title), xbmcName)); progressDialog.setMessage(getResources().getString(R.string.wizard_connecting_to_xbmc_message)); progressDialog.setCancelable(false); progressDialog.setIndeterminate(true); progressDialog.setOnShowListener(new DialogInterface.OnShowListener() { @Override public void onShow(DialogInterface dialog) { // Let's ping the host through HTTP chainCallCheckHttpConnection(checkedHostInfo); } }); progressDialog.show(); } private void chainCallCheckHttpConnection(final HostInfo hostInfo) { // Let's ping the host through HTTP final HostConnection hostConnection = new HostConnection(hostInfo); hostConnection.setProtocol(HostConnection.PROTOCOL_HTTP); final JSONRPC.Ping httpPing = new JSONRPC.Ping(); httpPing.execute(hostConnection, new ApiCallback() { @Override public void onSuccess(String result) { LogUtils.LOGD(TAG, "Successfully connected to new host through HTTP."); // Great, we managed to connect through HTTP, let's check through tcp if (hostInfo.getProtocol() == HostConnection.PROTOCOL_TCP) { chainCallCheckTcpConnection(hostConnection, hostInfo); } else { // No TCP, check EventServer hostConnection.disconnect(); chainCallCheckEventServerConnection(hostInfo); } } @Override public void onError(int errorCode, String description) { // Couldn't connect through HTTP, abort, and initialize checkedHostInfo hostConnection.disconnect(); hostConnectionError(errorCode, description); } }, handler); } private void chainCallCheckTcpConnection(final HostConnection hostConnection, final HostInfo hostInfo) { final JSONRPC.Ping tcpPing = new JSONRPC.Ping(); hostConnection.setProtocol(HostConnection.PROTOCOL_TCP); tcpPing.execute(hostConnection, new ApiCallback() { @Override public void onSuccess(String result) { // Great, we managed to connect through HTTP and TCP LogUtils.LOGD(TAG, "Successfully connected to new host through TCP."); hostConnection.disconnect(); // Check EventServer chainCallCheckEventServerConnection(hostInfo); } @Override public void onError(int errorCode, String description) { // We only managed to connect through HTTP, revert checkedHostInfo to use HTTP LogUtils.LOGD(TAG, "Couldn't connect to host through TCP. Message: " + description); hostConnection.disconnect(); hostInfo.setProtocol(HostConnection.PROTOCOL_HTTP); // Check EventServer chainCallCheckEventServerConnection(hostInfo); } }, handler); } private void chainCallCheckEventServerConnection(final HostInfo hostInfo) { if (hostInfo.getUseEventServer()) { EventServerConnection.testEventServerConnection( hostInfo, new EventServerConnection.EventServerConnectionCallback() { @Override public void OnConnectResult(boolean success) { LogUtils.LOGD(TAG, "Check ES connection: " + success); if (success) { chainCallCheckKodiVersion(hostInfo); } else { hostInfo.setUseEventServer(false); chainCallCheckKodiVersion(hostInfo); } } }, handler); } else { chainCallCheckKodiVersion(hostInfo); } } private void chainCallCheckKodiVersion(final HostInfo hostInfo) { final HostConnection hostConnection = new HostConnection(hostInfo); hostConnection.setProtocol(HostConnection.PROTOCOL_HTTP); final Application.GetProperties getProperties = new Application.GetProperties(Application.GetProperties.VERSION); getProperties.execute(hostConnection, new ApiCallback() { @Override public void onSuccess(ApplicationType.PropertyValue result) { LogUtils.LOGD(TAG, "Successfully checked Kodi version."); hostInfo.setKodiVersionMajor(result.version.major); hostInfo.setKodiVersionMinor(result.version.minor); hostInfo.setKodiVersionRevision(result.version.revision); hostInfo.setKodiVersionTag(result.version.tag); hostConnection.disconnect(); hostConnectionChecked(hostInfo); } @Override public void onError(int errorCode, String description) { // Couldn't get Kodi version... Odd, but let's proceed anyway with the defaults hostConnection.disconnect(); hostConnectionChecked(hostInfo); } }, handler); } /** * The connection was checked, and hostInfo has all the correct parameters to communicate * with it * @param hostInfo {@link HostInfo} to add */ private void hostConnectionChecked(final HostInfo hostInfo) { // Let's get the MAC Address, if we don't have one if (TextUtils.isEmpty(hostInfo.getMacAddress())) { new Thread(new Runnable() { @Override public void run() { String localMacAddress = NetUtils.getMacAddress(hostInfo.getAddress()); hostInfo.setMacAddress(localMacAddress); handler.post(new Runnable() { @Override public void run() { if (isAdded()) { progressDialog.dismiss(); listener.onHostManualConfigurationNext(hostInfo); } } }); } }).start(); } else { // Mac address was supplied if (isAdded()) { progressDialog.dismiss(); listener.onHostManualConfigurationNext(hostInfo); } } } /** * Treats errors occurred during the connection check * @param errorCode Error code * @param description Description */ private void hostConnectionError(int errorCode, String description) { if (!isAdded()) return; progressDialog.dismiss(); LogUtils.LOGD(TAG, "An error occurred during connection testint. Message: " + description); switch (errorCode) { case ApiException.HTTP_RESPONSE_CODE_UNAUTHORIZED: String username = xbmcUsernameEditText.getText().toString(), password = xbmcPasswordEditText.getText().toString(); int messageResourceId; if (TextUtils.isEmpty(username) || TextUtils.isEmpty(password)) { messageResourceId = R.string.wizard_empty_authentication; } else { messageResourceId = R.string.wizard_incorrect_authentication; } Toast.makeText(getActivity(), messageResourceId, Toast.LENGTH_SHORT).show(); xbmcUsernameEditText.requestFocus(); break; default: Toast.makeText(getActivity(), R.string.wizard_error_connecting, Toast.LENGTH_SHORT).show(); break; } } }