mirror of
https://github.com/emanuele-f/PCAPdroid.git
synced 2026-06-16 21:10:57 +08:00
394 lines
13 KiB
Java
394 lines
13 KiB
Java
/*
|
|
* This file is part of PCAPdroid.
|
|
*
|
|
* PCAPdroid is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* PCAPdroid is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with PCAPdroid. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Copyright 2020 - Emanuele Faranda
|
|
*/
|
|
|
|
package com.emanuelef.remote_capture;
|
|
|
|
import android.annotation.TargetApi;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.SharedPreferences;
|
|
import android.content.pm.PackageManager;
|
|
import android.net.ConnectivityManager;
|
|
import android.net.VpnService;
|
|
import android.os.Build;
|
|
import android.os.Bundle;
|
|
import android.os.ParcelFileDescriptor;
|
|
import android.util.Log;
|
|
import android.widget.Toast;
|
|
|
|
import androidx.localbroadcastmanager.content.LocalBroadcastManager;
|
|
import androidx.preference.PreferenceManager;
|
|
|
|
import java.io.IOException;
|
|
import java.net.InetSocketAddress;
|
|
|
|
public class CaptureService extends VpnService implements Runnable {
|
|
private static final String TAG = "CaptureService";
|
|
private static final String VpnSessionName = "PCAPdroid VPN";
|
|
private ParcelFileDescriptor mParcelFileDescriptor = null;
|
|
private Thread mThread;
|
|
private String vpn_ipv4;
|
|
private String vpn_dns;
|
|
private String public_dns;
|
|
private String collector_address;
|
|
private String tls_proxy_address;
|
|
private Prefs.DumpMode dump_mode;
|
|
private boolean tls_decryption_enabled;
|
|
private int collector_port;
|
|
private int http_server_port;
|
|
private int tls_proxy_port;
|
|
private long last_bytes;
|
|
private static CaptureService INSTANCE;
|
|
private String app_filter;
|
|
private HTTPServer mHttpServer;
|
|
|
|
/* The IP address of the virtual network interface */
|
|
public static final String VPN_IP_ADDRESS = "10.215.173.1";
|
|
|
|
/* The DNS server IP address to use to internally analyze the DNS requests.
|
|
* It must be in the same subnet of the VPN network interface.
|
|
* After the analysis, requests will be routed to the primary DNS server. */
|
|
public static final String VPN_VIRTUAL_DNS_SERVER = "10.215.173.2";
|
|
|
|
public static final String ACTION_TRAFFIC_STATS_UPDATE = "traffic_stats_update";
|
|
public static final String ACTION_CONNECTIONS_DUMP = "connections_dump";
|
|
public static final String ACTION_STATS_DUMP = "stats_dump";
|
|
public static final String TRAFFIC_STATS_UPDATE_SENT_BYTES = "sent_bytes";
|
|
public static final String TRAFFIC_STATS_UPDATE_RCVD_BYTES = "rcvd_bytes";
|
|
public static final String TRAFFIC_STATS_UPDATE_SENT_PKTS = "sent_pkts";
|
|
public static final String TRAFFIC_STATS_UPDATE_RCVD_PKTS = "rcvd_pkts";
|
|
|
|
public static final String ACTION_SERVICE_STATUS = "service_status";
|
|
public static final String SERVICE_STATUS_KEY = "status";
|
|
public static final String SERVICE_STATUS_STARTED = "started";
|
|
public static final String SERVICE_STATUS_STOPPED = "stopped";
|
|
|
|
static {
|
|
/* Load native library */
|
|
System.loadLibrary("vpnproxy-jni");
|
|
}
|
|
|
|
@Override
|
|
public void onCreate() {
|
|
Log.d(CaptureService.TAG, "onCreate");
|
|
INSTANCE = this;
|
|
super.onCreate();
|
|
}
|
|
|
|
@Override
|
|
public int onStartCommand(Intent intent, int flags, int startId) {
|
|
Context app_ctx = getApplicationContext();
|
|
|
|
if (intent == null) {
|
|
Log.d(CaptureService.TAG, "NULL intent onStartCommand");
|
|
return super.onStartCommand(null, flags, startId);
|
|
}
|
|
|
|
Log.d(CaptureService.TAG, "onStartCommand");
|
|
|
|
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(this);
|
|
Bundle settings = intent.getBundleExtra("settings");
|
|
|
|
if (settings == null) {
|
|
Log.e(CaptureService.TAG, "NULL settings");
|
|
return super.onStartCommand(null, flags, startId);
|
|
}
|
|
|
|
// Retrieve Configuration
|
|
public_dns = Utils.getDnsServer(app_ctx);
|
|
vpn_dns = VPN_VIRTUAL_DNS_SERVER;
|
|
vpn_ipv4 = VPN_IP_ADDRESS;
|
|
app_filter = settings.getString(Prefs.PREF_APP_FILTER);
|
|
|
|
collector_address = Prefs.getCollectorIp(prefs);
|
|
collector_port = Prefs.getCollectorPort(prefs);
|
|
http_server_port = Prefs.getHttpServerPort(prefs);
|
|
tls_decryption_enabled = Prefs.getTlsDecryptionEnabled(prefs);
|
|
tls_proxy_address = Prefs.getTlsProxyAddress(prefs);
|
|
tls_proxy_port = Prefs.getTlsProxyPort(prefs);
|
|
dump_mode = Prefs.getDumpMode(prefs);
|
|
last_bytes = 0;
|
|
|
|
if(dump_mode == Prefs.DumpMode.HTTP_SERVER) {
|
|
if (mHttpServer == null)
|
|
mHttpServer = new HTTPServer(app_ctx, http_server_port);
|
|
|
|
try {
|
|
mHttpServer.startConnections();
|
|
} catch (IOException e) {
|
|
Log.e(CaptureService.TAG, "Could not start the HTTP server");
|
|
e.printStackTrace();
|
|
}
|
|
} else
|
|
mHttpServer = null;
|
|
|
|
Log.i(TAG, "Using DNS server " + public_dns);
|
|
|
|
// VPN
|
|
/* In order to see the DNS packets into the VPN we must set an internal address as the DNS
|
|
* server. */
|
|
Builder builder = new Builder()
|
|
.addAddress(vpn_ipv4, 30) // using a random IP as an address is needed
|
|
.addRoute("0.0.0.0", 1)
|
|
.addRoute("128.0.0.0", 1)
|
|
.addDnsServer(vpn_dns);
|
|
|
|
if(app_filter != null) {
|
|
Log.d(TAG, "Setting app filter: " + app_filter);
|
|
|
|
try {
|
|
builder.addAllowedApplication(app_filter);
|
|
} catch (PackageManager.NameNotFoundException e) {
|
|
String msg = String.format(getResources().getString(R.string.app_not_found), app_filter);
|
|
Toast.makeText(this, msg, Toast.LENGTH_SHORT).show();
|
|
return super.onStartCommand(intent, flags, startId);
|
|
}
|
|
}
|
|
|
|
try {
|
|
mParcelFileDescriptor = builder.setSession(CaptureService.VpnSessionName).establish();
|
|
} catch (IllegalArgumentException | IllegalStateException e) {
|
|
Utils.showToast(this, R.string.vpn_setup_failed);
|
|
return super.onStartCommand(intent, flags, startId);
|
|
}
|
|
|
|
// Stop the previous session by interrupting the thread.
|
|
if (mThread != null) {
|
|
mThread.interrupt();
|
|
}
|
|
|
|
// Start a new session by creating a new thread.
|
|
mThread = new Thread(this, "CaptureService Thread");
|
|
mThread.start();
|
|
return START_STICKY;
|
|
//return super.onStartCommand(intent, flags, startId);
|
|
}
|
|
|
|
@Override
|
|
public void onRevoke() {
|
|
Log.d(CaptureService.TAG, "onRevoke");
|
|
stop();
|
|
|
|
super.onRevoke();
|
|
}
|
|
|
|
@Override
|
|
public void onDestroy() {
|
|
Log.d(CaptureService.TAG, "onDestroy");
|
|
stop();
|
|
INSTANCE = null;
|
|
|
|
if(mThread != null) {
|
|
mThread.interrupt();
|
|
}
|
|
if(mHttpServer != null)
|
|
mHttpServer.stop();
|
|
|
|
super.onDestroy();
|
|
}
|
|
|
|
private void stop() {
|
|
stopPacketLoop();
|
|
|
|
while((mThread != null) && (mThread.isAlive())) {
|
|
try {
|
|
Log.d(TAG, "Joining native thread...");
|
|
mThread.join();
|
|
} catch (InterruptedException e) {
|
|
Log.e(TAG, "Joining native thread failed");
|
|
}
|
|
}
|
|
|
|
mThread = null;
|
|
|
|
if(mParcelFileDescriptor != null) {
|
|
try {
|
|
mParcelFileDescriptor.close();
|
|
} catch (IOException e) {
|
|
Toast.makeText(this, "Stopping VPN failed", Toast.LENGTH_SHORT).show();
|
|
}
|
|
mParcelFileDescriptor = null;
|
|
}
|
|
|
|
if(mHttpServer != null)
|
|
mHttpServer.endConnections();
|
|
// NOTE: do not destroy the mHttpServer, let it terminate the active connections
|
|
}
|
|
|
|
/* Check if the VPN service was launched */
|
|
public static boolean isServiceActive() {
|
|
return((INSTANCE != null) &&
|
|
(INSTANCE.mParcelFileDescriptor != null));
|
|
}
|
|
|
|
public static String getAppFilter() {
|
|
return((INSTANCE != null) ? INSTANCE.app_filter : null);
|
|
}
|
|
|
|
public static long getBytes() {
|
|
return((INSTANCE != null) ? INSTANCE.last_bytes : 0);
|
|
}
|
|
|
|
public static String getCollectorAddress() {
|
|
return((INSTANCE != null) ? INSTANCE.collector_address : "");
|
|
}
|
|
|
|
public static int getCollectorPort() {
|
|
return((INSTANCE != null) ? INSTANCE.collector_port : 0);
|
|
}
|
|
|
|
public static int getHTTPServerPort() {
|
|
return((INSTANCE != null) ? INSTANCE.http_server_port : 0);
|
|
}
|
|
|
|
public static Prefs.DumpMode getDumpMode() {
|
|
return((INSTANCE != null) ? INSTANCE.dump_mode : Prefs.DumpMode.NONE);
|
|
}
|
|
|
|
/* Stop a running VPN service */
|
|
public static void stopService() {
|
|
if (INSTANCE != null)
|
|
INSTANCE.stop();
|
|
}
|
|
|
|
@Override
|
|
public void run() {
|
|
if(mParcelFileDescriptor != null) {
|
|
int fd = mParcelFileDescriptor.getFd();
|
|
|
|
if(fd > 0)
|
|
runPacketLoop(fd, this, Build.VERSION.SDK_INT);
|
|
else
|
|
Log.e(TAG, "Invalid VPN fd: " + fd);
|
|
}
|
|
}
|
|
|
|
/* The following methods are called from native code */
|
|
|
|
public String getVpnIPv4() {
|
|
return(vpn_ipv4);
|
|
}
|
|
|
|
public String getVpnDns() {
|
|
return(vpn_dns);
|
|
}
|
|
|
|
public String getPublicDns() {
|
|
return(public_dns);
|
|
}
|
|
|
|
public String getPcapCollectorAddress() {
|
|
return(collector_address);
|
|
}
|
|
|
|
public int getPcapCollectorPort() {
|
|
return(collector_port);
|
|
}
|
|
|
|
public String getTlsProxyAddress() { return(tls_proxy_address); }
|
|
|
|
public int getTlsDecryptionEnabled() { return tls_decryption_enabled ? 1 : 0; }
|
|
|
|
public int getTlsProxyPort() { return(tls_proxy_port); }
|
|
|
|
// returns 1 if dumpPcapData should be called
|
|
public int dumpPcapToJava() {
|
|
return((mHttpServer != null) ? 1 : 0);
|
|
}
|
|
|
|
public int dumpPcapToUdp() {
|
|
return((dump_mode == Prefs.DumpMode.UDP_EXPORTER) ? 1 : 0);
|
|
}
|
|
|
|
// from NetGuard
|
|
@TargetApi(Build.VERSION_CODES.Q)
|
|
public int getUidQ(int version, int protocol, String saddr, int sport, String daddr, int dport) {
|
|
if (protocol != 6 /* TCP */ && protocol != 17 /* UDP */)
|
|
return -1;
|
|
|
|
ConnectivityManager cm = (ConnectivityManager) getSystemService(CONNECTIVITY_SERVICE);
|
|
if (cm == null)
|
|
return -1;
|
|
|
|
InetSocketAddress local = new InetSocketAddress(saddr, sport);
|
|
InetSocketAddress remote = new InetSocketAddress(daddr, dport);
|
|
|
|
Log.i(TAG, "Get uid local=" + local + " remote=" + remote);
|
|
int uid = cm.getConnectionOwnerUid(protocol, local, remote);
|
|
Log.i(TAG, "Get uid=" + uid);
|
|
return uid;
|
|
}
|
|
|
|
public void sendCaptureStats(long sent_bytes, long rcvd_bytes, int sent_pkts, int rcvd_pkts) {
|
|
Intent intent = new Intent(ACTION_TRAFFIC_STATS_UPDATE);
|
|
|
|
intent.putExtra(TRAFFIC_STATS_UPDATE_SENT_BYTES, sent_bytes);
|
|
intent.putExtra(TRAFFIC_STATS_UPDATE_RCVD_BYTES, rcvd_bytes);
|
|
intent.putExtra(TRAFFIC_STATS_UPDATE_SENT_PKTS, sent_pkts);
|
|
intent.putExtra(TRAFFIC_STATS_UPDATE_RCVD_PKTS, rcvd_pkts);
|
|
|
|
last_bytes = sent_bytes + rcvd_bytes;
|
|
|
|
LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
|
|
}
|
|
|
|
public void sendConnectionsDump(ConnDescriptor connections[]) {
|
|
Log.d(TAG, "sendConnectionsDump(" + connections.length + " connections)");
|
|
|
|
Bundle bundle = new Bundle();
|
|
bundle.putSerializable("value", connections);
|
|
Intent intent = new Intent(ACTION_CONNECTIONS_DUMP);
|
|
intent.putExtras(bundle);
|
|
|
|
LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
|
|
}
|
|
|
|
public void sendStatsDump(VPNStats stats) {
|
|
Log.d(TAG, "sendStatsDump");
|
|
|
|
Bundle bundle = new Bundle();
|
|
bundle.putSerializable("value", stats);
|
|
Intent intent = new Intent(ACTION_STATS_DUMP);
|
|
intent.putExtras(bundle);
|
|
|
|
LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
|
|
}
|
|
|
|
private void sendServiceStatus(String cur_status) {
|
|
Intent intent = new Intent(ACTION_SERVICE_STATUS);
|
|
intent.putExtra(SERVICE_STATUS_KEY, cur_status);
|
|
LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
|
|
}
|
|
|
|
public String getApplicationByUid(int uid) {
|
|
return(getPackageManager().getNameForUid(uid));
|
|
}
|
|
|
|
/* Exports a PCAP data chunk */
|
|
public void dumpPcapData(byte[] data) {
|
|
if(mHttpServer != null)
|
|
mHttpServer.pushData(data);
|
|
}
|
|
|
|
public static native void runPacketLoop(int fd, CaptureService vpn, int sdk);
|
|
public static native void stopPacketLoop();
|
|
public static native void askConnectionsDump();
|
|
public static native void askStatsDump();
|
|
}
|