PCAPdroid/app/src/main/java/com/emanuelef/remote_capture/CaptureService.java
2019-11-08 00:33:31 +01:00

277 lines
9.0 KiB
Java

/*
This file is part of RemoteCapture.
RemoteCapture 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.
RemoteCapture 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 RemoteCapture. If not, see <http://www.gnu.org/licenses/>.
Copyright 2019 by Emanuele Faranda
*/
package com.emanuelef.remote_capture;
import android.annotation.TargetApi;
import android.content.Intent;
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 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 = "Remote Capture 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 boolean capture_unknown_app_traffic;
private int collector_port;
private int uid_filter;
private long last_bytes;
private static CaptureService INSTANCE;
public static final String ACTION_TRAFFIC_STATS_UPDATE = "traffic_stats_update";
public static final String ACTION_CONNECTIONS_DUMP = "connections_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) {
if (intent == null) {
Log.d(CaptureService.TAG, "NULL intent onStartCommand");
return super.onStartCommand(intent, flags, startId);
}
Log.d(CaptureService.TAG, "onStartCommand");
Bundle settings = intent.getBundleExtra("settings");
// retrieve settings
assert settings != null;
public_dns = settings.getString("dns_server");
vpn_dns = "10.215.173.2";
vpn_ipv4 = "10.215.173.1";
collector_address = settings.getString(Prefs.PREF_COLLECTOR_IP_KEY);
collector_port = settings.getInt(Prefs.PREF_COLLECTOR_PORT_KEY);;
uid_filter = settings.getInt(Prefs.PREF_UID_FILTER);
capture_unknown_app_traffic = settings.getBoolean(Prefs.PREF_CAPTURE_UNKNOWN_APP_TRAFFIC);
last_bytes = 0;
// 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)
.addDnsServer(vpn_dns);
try {
mParcelFileDescriptor = builder.setSession(CaptureService.VpnSessionName).establish();
} catch (IllegalArgumentException | IllegalStateException e) {
Toast.makeText(this, "VPN setup failed", Toast.LENGTH_SHORT).show();
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();
}
super.onDestroy();
}
private void stop() {
if(mParcelFileDescriptor != null) {
try {
mParcelFileDescriptor.close();
} catch (IOException e) {
Toast.makeText(this, "Stopping VPN failed", Toast.LENGTH_SHORT).show();
}
mParcelFileDescriptor = null;
}
}
/* Check if the VPN service was launched */
public static boolean isServiceActive() {
return((INSTANCE != null) &&
(INSTANCE.mParcelFileDescriptor != null));
}
public static int getUidFilter() {
return((INSTANCE != null) ? INSTANCE.uid_filter : -1);
}
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);
}
/* Stop a running VPN service */
public static void stopService() {
if (INSTANCE != null) {
stopPacketLoop();
INSTANCE.stop();
}
}
@Override
public void run() {
runPacketLoop(mParcelFileDescriptor.detachFd(), this, Build.VERSION.SDK_INT);
}
/* 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 int getPcapUidFilter() {
return(uid_filter);
}
public int getCaptureUnknownTraffic() {
return(capture_unknown_app_traffic ? 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);
}
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));
}
public static native void runPacketLoop(int fd, CaptureService vpn, int sdk);
public static native void stopPacketLoop();
}