Android VPNService简述

前言

Android为开发人员提供了用于创建虚拟专用网(VPN)解决方案的API,也就是VPNService,使用VPNService可以为Android设备开发和测试自己的VPN客户端。本文将简单介绍VPNService的用法。

一、概述

Android包含一个内置的VPN客户端(也称作传统VPN)。Android4.0(API 14)引入了VPNService的API,以便应用程序开发人员可以提供自己的VPN解决方案。

Android中包括有一些VPN连接的UI组件:

在VPN应用程序首次激活之前,系统会显示一个连接请求对话框,该对话框提示使用该设备的人员确认他们信任的VPN,主要包含有如下的UI提示:

  1. 在VPN应用程序首次激活之前,系统会显示一个连接请求对话框,该对话框会提示使用该设备的人员确认是否要信任此应用并接受VPN请求,如下图所示:
VPNService1.jpg
  1. 在VPN程序连接中,在状态栏会存在一个信息推送,显示秘钥图标。用于指示当前VPN活动的连接。如下图所示:
VPNService2.jpg

二、VPNService基本使用

VPNService是Service的一个子类,因此我们使用VPNService主要包括如下几个步骤:

  1. 新建一个继承自VPNService的子类Service。
  2. AndroidManifest.xml文件中注册子类Service,并同时申请权限。
  3. 启动子类Service。

下面将对如上步骤进行逐一说明。

1. 创建VPNService的子类Service。

新建类TestVPNService继承自VPNService。

public class TestVPNService extends VpnService {
    private static final String TAG = "NetGuard.Service";
    private static final String EXTRA_COMMAND = "Command";
    private ParcelFileDescriptor vpn = null;
    public static final int START = 1;
    public static final int RELOAD = 2;
    public static final int STOP = 3;
    @Override
    public void onCreate() {
        // Listen for connectivity updates
        IntentFilter ifConnectivity = new IntentFilter();
        ifConnectivity.addAction(ConnectivityManager.CONNECTIVITY_ACTION);
        super.onCreate();
    }
    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        // Get command
        int cmd = intent.getIntExtra(EXTRA_COMMAND, RELOAD);
        Log.d(TAG, "执行:" + cmd);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            setForegroundService();
        }
        ...
    }
}

VPNService的创建需要通过Builder去创建及配置,最终生成一个ParcelFileDescriptor对象,并可以在Builder中进行代理配置,示例代码如下:

private ParcelFileDescriptor vpn = null;
Builder builder = new Builder();
        String[] appPackages = {
                "com.example.plugindemo1"
        };

        PackageManager packageManager = getPackageManager();

        builder.setSession(getString(R.string.app_name));
        
        builder.addAddress("10.1.10.1", 32);
        builder.addAddress("fd00:1:fd00:1:fd00:1:fd00:1", 128);
        builder.addRoute("0.0.0.0", 0);
        builder.addRoute("0:0:0:0:0:0:0:0", 0);
        /**
         * builder.setMtu(int mut)//设置读写操作时最大缓存
         *        .setSession(String session)//设置该次服务名称,服务启动后可在手机设置界面查看
         *        .addAddress(String address, int port)//设置虚拟主机地址和端口
         *        .addRoute(String address, int port)//设置允许通过的路由
         *        .addDnsServer(String address)//添加域名服务器
         *        .addAllowedApplication(String name)//添加允许访问连接的程序
         *        .setConfigureIntent(PendingIntent intent);//设置配置启动项
         */

        for (String appPackage: appPackages) {
            try {
                packageManager.getPackageInfo(appPackage, 0);
                builder.addAllowedApplication(appPackage);
            } catch (PackageManager.NameNotFoundException e) {
                // The app isn't installed.
                Logger.get().d("添加允许的VPN应用出错,未找到");
                e.printStackTrace();
            }
        }

最后通过调用如下代码就可以打开代理了。

vpn = builder.establish();
2. 权限申请

首先需要在AndroidManifest.xml配置文件中添加基本的网络权限:

   <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
    <uses-permission android:name="android.permission.INTERNET" />

除此之外,还需要对之前定义的Service进行声明并配置其权限:

        <service
            android:name=".TestVPNService"
            android:permission="android.permission.BIND_VPN_SERVICE">
            <intent-filter>
                <action android:name="android.net.VpnService" />
            </intent-filter>
            <meta-data
                android:name="android.net.VpnService.SUPPORTS_ALWAYS_ON"
                android:value="true" />
        </service>
3. 启动VPNService

启动自定义的VPNService首先需要调用prepare()方法获取intent,然后在启动Service,示例如下:

    private fun startVPN(){
        var intent1 = TestVPNService.prepare(this)
        ContextCompat.startForegroundService(this, intent1)
    }

总结

由于VPNService在平常的很少被使用到,所以本文仅仅是对其的基本使用做简单介绍,本人也是在调研网络代理时才了解了此内容。需要进阶使用的朋友可以查看官方文档。

参考文章

官方使用文档

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容