2014-10-22 18:38:44 +02:00
|
|
|
package eu.siacs.conversations.http;
|
|
|
|
|
2021-03-22 18:03:25 +01:00
|
|
|
import android.os.Build;
|
2018-03-18 10:30:15 +01:00
|
|
|
import android.util.Log;
|
|
|
|
|
2015-07-10 14:14:45 +02:00
|
|
|
import org.apache.http.conn.ssl.StrictHostnameVerifier;
|
|
|
|
|
2021-03-22 10:39:53 +01:00
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.InputStream;
|
2015-11-28 20:11:38 +01:00
|
|
|
import java.net.InetAddress;
|
|
|
|
import java.net.InetSocketAddress;
|
|
|
|
import java.net.Proxy;
|
2021-03-19 14:57:15 +01:00
|
|
|
import java.net.UnknownHostException;
|
2015-07-10 14:14:45 +02:00
|
|
|
import java.security.KeyManagementException;
|
|
|
|
import java.security.NoSuchAlgorithmException;
|
2019-01-28 10:39:01 +01:00
|
|
|
import java.util.ArrayList;
|
2014-10-22 18:38:44 +02:00
|
|
|
import java.util.List;
|
2021-02-22 09:24:41 +01:00
|
|
|
import java.util.concurrent.Executor;
|
|
|
|
import java.util.concurrent.Executors;
|
2021-03-19 14:57:15 +01:00
|
|
|
import java.util.concurrent.TimeUnit;
|
2014-10-22 18:38:44 +02:00
|
|
|
|
2015-07-10 14:14:45 +02:00
|
|
|
import javax.net.ssl.SSLSocketFactory;
|
|
|
|
import javax.net.ssl.X509TrustManager;
|
|
|
|
|
2021-04-30 10:54:36 +02:00
|
|
|
import eu.siacs.conversations.BuildConfig;
|
2018-03-18 10:30:15 +01:00
|
|
|
import eu.siacs.conversations.Config;
|
|
|
|
import eu.siacs.conversations.entities.Account;
|
2014-10-22 18:38:44 +02:00
|
|
|
import eu.siacs.conversations.entities.Message;
|
|
|
|
import eu.siacs.conversations.services.AbstractConnectionManager;
|
|
|
|
import eu.siacs.conversations.services.XmppConnectionService;
|
2016-11-20 00:39:01 +01:00
|
|
|
import eu.siacs.conversations.utils.TLSSocketFactory;
|
2021-03-19 14:57:15 +01:00
|
|
|
import okhttp3.HttpUrl;
|
|
|
|
import okhttp3.OkHttpClient;
|
2021-03-22 10:39:53 +01:00
|
|
|
import okhttp3.Request;
|
|
|
|
import okhttp3.ResponseBody;
|
2014-10-22 18:38:44 +02:00
|
|
|
|
|
|
|
public class HttpConnectionManager extends AbstractConnectionManager {
|
|
|
|
|
2019-01-28 10:39:01 +01:00
|
|
|
private final List<HttpDownloadConnection> downloadConnections = new ArrayList<>();
|
|
|
|
private final List<HttpUploadConnection> uploadConnections = new ArrayList<>();
|
|
|
|
|
2021-02-22 09:24:41 +01:00
|
|
|
public static final Executor EXECUTOR = Executors.newFixedThreadPool(4);
|
|
|
|
|
2021-04-30 10:54:36 +02:00
|
|
|
public static final OkHttpClient OK_HTTP_CLIENT;
|
|
|
|
|
|
|
|
static {
|
|
|
|
OK_HTTP_CLIENT = new OkHttpClient.Builder()
|
|
|
|
.addInterceptor(chain -> {
|
|
|
|
final Request original = chain.request();
|
|
|
|
final Request modified = original.newBuilder()
|
|
|
|
.header("User-Agent", getUserAgent())
|
|
|
|
.build();
|
|
|
|
return chain.proceed(modified);
|
|
|
|
})
|
|
|
|
.build();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public static String getUserAgent() {
|
|
|
|
return String.format("%s/%s", BuildConfig.APP_NAME, BuildConfig.VERSION_NAME);
|
|
|
|
}
|
2021-04-18 15:46:37 +02:00
|
|
|
|
2019-01-28 10:39:01 +01:00
|
|
|
public HttpConnectionManager(XmppConnectionService service) {
|
|
|
|
super(service);
|
|
|
|
}
|
|
|
|
|
2021-03-19 14:57:15 +01:00
|
|
|
public static Proxy getProxy() {
|
2021-03-22 18:03:25 +01:00
|
|
|
final InetAddress localhost;
|
2021-03-19 14:57:15 +01:00
|
|
|
try {
|
2021-03-22 18:03:25 +01:00
|
|
|
localhost = InetAddress.getByAddress(new byte[]{127, 0, 0, 1});
|
2021-03-19 14:57:15 +01:00
|
|
|
} catch (final UnknownHostException e) {
|
2021-04-18 15:46:37 +02:00
|
|
|
throw new IllegalStateException(e);
|
|
|
|
}
|
2021-03-22 18:03:25 +01:00
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
|
|
|
|
return new Proxy(Proxy.Type.SOCKS, new InetSocketAddress(localhost, 9050));
|
|
|
|
} else {
|
|
|
|
return new Proxy(Proxy.Type.HTTP, new InetSocketAddress(localhost, 8118));
|
2021-03-19 14:57:15 +01:00
|
|
|
}
|
2019-01-28 10:39:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public void createNewDownloadConnection(Message message) {
|
|
|
|
this.createNewDownloadConnection(message, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void createNewDownloadConnection(final Message message, boolean interactive) {
|
|
|
|
synchronized (this.downloadConnections) {
|
2021-04-18 15:46:37 +02:00
|
|
|
for (HttpDownloadConnection connection : this.downloadConnections) {
|
2019-01-28 10:39:01 +01:00
|
|
|
if (connection.getMessage() == message) {
|
|
|
|
Log.d(Config.LOGTAG, message.getConversation().getAccount().getJid().asBareJid() + ": download already in progress");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
final HttpDownloadConnection connection = new HttpDownloadConnection(message, this);
|
|
|
|
connection.init(interactive);
|
|
|
|
this.downloadConnections.add(connection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void createNewUploadConnection(final Message message, boolean delay) {
|
|
|
|
synchronized (this.uploadConnections) {
|
|
|
|
for (HttpUploadConnection connection : this.uploadConnections) {
|
|
|
|
if (connection.getMessage() == message) {
|
|
|
|
Log.d(Config.LOGTAG, message.getConversation().getAccount().getJid().asBareJid() + ": upload already in progress");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
HttpUploadConnection connection = new HttpUploadConnection(message, Method.determine(message.getConversation().getAccount()), this);
|
|
|
|
connection.init(delay);
|
|
|
|
this.uploadConnections.add(connection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void finishConnection(HttpDownloadConnection connection) {
|
|
|
|
synchronized (this.downloadConnections) {
|
|
|
|
this.downloadConnections.remove(connection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void finishUploadConnection(HttpUploadConnection httpUploadConnection) {
|
|
|
|
synchronized (this.uploadConnections) {
|
|
|
|
this.uploadConnections.remove(httpUploadConnection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-19 14:57:15 +01:00
|
|
|
OkHttpClient buildHttpClient(final HttpUrl url, final Account account, boolean interactive) {
|
2021-04-18 15:46:37 +02:00
|
|
|
return buildHttpClient(url, account, 30, interactive);
|
|
|
|
}
|
|
|
|
|
|
|
|
OkHttpClient buildHttpClient(final HttpUrl url, final Account account, int readTimeout, boolean interactive) {
|
2021-03-19 14:57:15 +01:00
|
|
|
final String slotHostname = url.host();
|
|
|
|
final boolean onionSlot = slotHostname.endsWith(".onion");
|
2021-04-18 15:46:37 +02:00
|
|
|
final OkHttpClient.Builder builder = OK_HTTP_CLIENT.newBuilder();
|
2021-03-19 14:57:15 +01:00
|
|
|
builder.writeTimeout(30, TimeUnit.SECONDS);
|
2021-04-18 15:46:37 +02:00
|
|
|
builder.readTimeout(readTimeout, TimeUnit.SECONDS);
|
2021-03-19 14:57:15 +01:00
|
|
|
setupTrustManager(builder, interactive);
|
|
|
|
if (mXmppConnectionService.useTorToConnect() || account.isOnion() || onionSlot) {
|
|
|
|
builder.proxy(HttpConnectionManager.getProxy()).build();
|
|
|
|
}
|
|
|
|
return builder.build();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setupTrustManager(final OkHttpClient.Builder builder, final boolean interactive) {
|
2019-01-28 10:39:01 +01:00
|
|
|
final X509TrustManager trustManager;
|
|
|
|
if (interactive) {
|
|
|
|
trustManager = mXmppConnectionService.getMemorizingTrustManager().getInteractive();
|
|
|
|
} else {
|
|
|
|
trustManager = mXmppConnectionService.getMemorizingTrustManager().getNonInteractive();
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
final SSLSocketFactory sf = new TLSSocketFactory(new X509TrustManager[]{trustManager}, mXmppConnectionService.getRNG());
|
2021-03-19 14:57:15 +01:00
|
|
|
builder.sslSocketFactory(sf, trustManager);
|
2021-04-30 09:53:19 +02:00
|
|
|
builder.hostnameVerifier(new StrictHostnameVerifier());
|
2019-01-28 10:39:01 +01:00
|
|
|
} catch (final KeyManagementException | NoSuchAlgorithmException ignored) {
|
|
|
|
}
|
|
|
|
}
|
2021-03-22 10:39:53 +01:00
|
|
|
|
|
|
|
public static InputStream open(final String url, final boolean tor) throws IOException {
|
|
|
|
return open(HttpUrl.get(url), tor);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static InputStream open(final HttpUrl httpUrl, final boolean tor) throws IOException {
|
2021-04-18 15:46:37 +02:00
|
|
|
final OkHttpClient.Builder builder = OK_HTTP_CLIENT.newBuilder();
|
2021-03-22 10:39:53 +01:00
|
|
|
if (tor) {
|
|
|
|
builder.proxy(HttpConnectionManager.getProxy()).build();
|
|
|
|
}
|
|
|
|
final OkHttpClient client = builder.build();
|
|
|
|
final Request request = new Request.Builder().get().url(httpUrl).build();
|
|
|
|
final ResponseBody body = client.newCall(request).execute().body();
|
|
|
|
if (body == null) {
|
|
|
|
throw new IOException("No response body found");
|
|
|
|
}
|
|
|
|
return body.byteStream();
|
|
|
|
}
|
2014-10-22 18:38:44 +02:00
|
|
|
}
|