package de.danoeh.antennapod.dialog; import android.app.Dialog; import android.content.Context; import android.content.res.TypedArray; import android.os.Build; import androidx.appcompat.app.AlertDialog; import androidx.core.content.ContextCompat; import android.text.Editable; import android.text.TextUtils; import android.text.TextWatcher; import android.util.Patterns; import android.view.View; import android.widget.AdapterView; import android.widget.ArrayAdapter; import android.widget.EditText; import android.widget.Spinner; import android.widget.TextView; import java.io.IOException; import java.net.InetSocketAddress; import java.net.Proxy; import java.net.SocketAddress; import java.util.ArrayList; import java.util.List; import java.util.Locale; import java.util.concurrent.TimeUnit; import de.danoeh.antennapod.R; import de.danoeh.antennapod.core.preferences.UserPreferences; import de.danoeh.antennapod.core.service.download.AntennapodHttpClient; import de.danoeh.antennapod.core.service.download.ProxyConfig; import io.reactivex.Completable; import io.reactivex.android.schedulers.AndroidSchedulers; import io.reactivex.disposables.Disposable; import io.reactivex.schedulers.Schedulers; import okhttp3.Credentials; import okhttp3.OkHttpClient; import okhttp3.Request; import okhttp3.Response; public class ProxyDialog { private final Context context; private AlertDialog dialog; private Spinner spType; private EditText etHost; private EditText etPort; private EditText etUsername; private EditText etPassword; private boolean testSuccessful = false; private TextView txtvMessage; private Disposable disposable; public ProxyDialog(Context context) { this.context = context; } public Dialog show() { View content = View.inflate(context, R.layout.proxy_settings, null); spType = content.findViewById(R.id.spType); dialog = new AlertDialog.Builder(context) .setTitle(R.string.pref_proxy_title) .setView(content) .setNegativeButton(R.string.cancel_label, null) .setPositiveButton(R.string.proxy_test_label, null) .show(); // To prevent cancelling the dialog on button click dialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener((view) -> { if (!testSuccessful) { dialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(false); test(); return; } String type = (String) spType.getSelectedItem(); ProxyConfig proxy; if (Proxy.Type.valueOf(type) == Proxy.Type.DIRECT) { proxy = ProxyConfig.direct(); } else { String host = etHost.getText().toString(); String port = etPort.getText().toString(); String username = etUsername.getText().toString(); if (TextUtils.isEmpty(username)) { username = null; } String password = etPassword.getText().toString(); if (TextUtils.isEmpty(password)) { password = null; } int portValue = 0; if (!TextUtils.isEmpty(port)) { portValue = Integer.parseInt(port); } if (Proxy.Type.valueOf(type) == Proxy.Type.SOCKS) { proxy = ProxyConfig.socks(host, portValue, username, password); } else { proxy = ProxyConfig.http(host, portValue, username, password); } } UserPreferences.setProxyConfig(proxy); AntennapodHttpClient.reinit(); dialog.dismiss(); }); List types = new ArrayList<>(); types.add(Proxy.Type.DIRECT.name()); types.add(Proxy.Type.HTTP.name()); if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { types.add(Proxy.Type.SOCKS.name()); } ArrayAdapter adapter = new ArrayAdapter<>(context, android.R.layout.simple_spinner_item, types); adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item); spType.setAdapter(adapter); ProxyConfig proxyConfig = UserPreferences.getProxyConfig(); spType.setSelection(adapter.getPosition(proxyConfig.type.name())); etHost = content.findViewById(R.id.etHost); if (!TextUtils.isEmpty(proxyConfig.host)) { etHost.setText(proxyConfig.host); } etHost.addTextChangedListener(requireTestOnChange); etPort = content.findViewById(R.id.etPort); if (proxyConfig.port > 0) { etPort.setText(String.valueOf(proxyConfig.port)); } etPort.addTextChangedListener(requireTestOnChange); etUsername = content.findViewById(R.id.etUsername); if (!TextUtils.isEmpty(proxyConfig.username)) { etUsername.setText(proxyConfig.username); } etUsername.addTextChangedListener(requireTestOnChange); etPassword = content.findViewById(R.id.etPassword); if (!TextUtils.isEmpty(proxyConfig.password)) { etPassword.setText(proxyConfig.password); } etPassword.addTextChangedListener(requireTestOnChange); if (proxyConfig.type == Proxy.Type.DIRECT) { enableSettings(false); setTestRequired(false); } spType.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() { @Override public void onItemSelected(AdapterView parent, View view, int position, long id) { enableSettings(position > 0); setTestRequired(position > 0); } @Override public void onNothingSelected(AdapterView parent) { enableSettings(false); } }); txtvMessage = content.findViewById(R.id.txtvMessage); checkValidity(); return dialog; } private final TextWatcher requireTestOnChange = new TextWatcher() { @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) {} @Override public void onTextChanged(CharSequence s, int start, int before, int count) {} @Override public void afterTextChanged(Editable s) { setTestRequired(true); } }; private void enableSettings(boolean enable) { etHost.setEnabled(enable); etPort.setEnabled(enable); etUsername.setEnabled(enable); etPassword.setEnabled(enable); } private boolean checkValidity() { boolean valid = true; if (spType.getSelectedItemPosition() > 0) { valid = checkHost(); } valid &= checkPort(); return valid; } private boolean checkHost() { String host = etHost.getText().toString(); if (host.length() == 0) { etHost.setError(context.getString(R.string.proxy_host_empty_error)); return false; } if (!"localhost".equals(host) && !Patterns.DOMAIN_NAME.matcher(host).matches()) { etHost.setError(context.getString(R.string.proxy_host_invalid_error)); return false; } return true; } private boolean checkPort() { int port = getPort(); if (port < 0 || port > 65535) { etPort.setError(context.getString(R.string.proxy_port_invalid_error)); return false; } return true; } private int getPort() { String port = etPort.getText().toString(); if (port.length() > 0) { try { return Integer.parseInt(port); } catch (NumberFormatException e) { // ignore } } return 0; } private void setTestRequired(boolean required) { if (required) { testSuccessful = false; dialog.getButton(AlertDialog.BUTTON_POSITIVE).setText(R.string.proxy_test_label); } else { testSuccessful = true; dialog.getButton(AlertDialog.BUTTON_POSITIVE).setText(android.R.string.ok); } dialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(true); } private void test() { if (disposable != null) { disposable.dispose(); } if (!checkValidity()) { setTestRequired(true); return; } TypedArray res = context.getTheme().obtainStyledAttributes(new int[] { android.R.attr.textColorPrimary }); int textColorPrimary = res.getColor(0, 0); res.recycle(); String checking = context.getString(R.string.proxy_checking); txtvMessage.setTextColor(textColorPrimary); txtvMessage.setText("{fa-circle-o-notch spin} " + checking); txtvMessage.setVisibility(View.VISIBLE); disposable = Completable.create(emitter -> { String type = (String) spType.getSelectedItem(); String host = etHost.getText().toString(); String port = etPort.getText().toString(); String username = etUsername.getText().toString(); String password = etPassword.getText().toString(); int portValue = 8080; if (!TextUtils.isEmpty(port)) { portValue = Integer.parseInt(port); } SocketAddress address = InetSocketAddress.createUnresolved(host, portValue); Proxy.Type proxyType = Proxy.Type.valueOf(type.toUpperCase(Locale.US)); OkHttpClient.Builder builder = AntennapodHttpClient.newBuilder() .connectTimeout(10, TimeUnit.SECONDS) .proxy(new Proxy(proxyType, address)); if (!TextUtils.isEmpty(username)) { builder.proxyAuthenticator((route, response) -> { String credentials = Credentials.basic(username, password); return response.request().newBuilder() .header("Proxy-Authorization", credentials) .build(); }); } OkHttpClient client = builder.build(); Request request = new Request.Builder().url("https://www.example.com").head().build(); try (Response response = client.newCall(request).execute()) { if (response.isSuccessful()) { emitter.onComplete(); } else { emitter.onError(new IOException(response.message())); } } catch (IOException e) { emitter.onError(e); } }) .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe( () -> { txtvMessage.setTextColor(ContextCompat.getColor(context, R.color.download_success_green)); String message = String.format("%s %s", "{fa-check}", context.getString(R.string.proxy_test_successful)); txtvMessage.setText(message); setTestRequired(false); }, error -> { error.printStackTrace(); txtvMessage.setTextColor(ContextCompat.getColor(context, R.color.download_failed_red)); String message = String.format("%s %s: %s", "{fa-close}", context.getString(R.string.proxy_test_failed), error.getMessage()); txtvMessage.setText(message); setTestRequired(true); } ); } }