2009-06-03 15:56:34 +00:00
|
|
|
/*
|
2009-06-19 11:45:38 +00:00
|
|
|
* Copyright (C) 2009, The Android Open Source Project
|
2009-06-03 15:56:34 +00:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package android.net.vpn;
|
|
|
|
|
2009-11-18 18:14:20 +00:00
|
|
|
import java.io.File;
|
|
|
|
|
2009-06-03 15:56:34 +00:00
|
|
|
import android.content.BroadcastReceiver;
|
|
|
|
import android.content.Context;
|
|
|
|
import android.content.Intent;
|
|
|
|
import android.content.IntentFilter;
|
|
|
|
import android.content.ServiceConnection;
|
2009-11-18 18:14:20 +00:00
|
|
|
import android.os.Environment;
|
|
|
|
import android.os.SystemProperties;
|
2009-06-03 15:56:34 +00:00
|
|
|
import android.util.Log;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The class provides interface to manage all VPN-related tasks, including:
|
|
|
|
* <ul>
|
|
|
|
* <li>The list of supported VPN types.
|
|
|
|
* <li>API's to start/stop the service of a particular type.
|
|
|
|
* <li>API's to start the settings activity.
|
|
|
|
* <li>API's to create a profile.
|
|
|
|
* <li>API's to register/unregister a connectivity receiver and the keys to
|
|
|
|
* access the fields in a connectivity broadcast event.
|
|
|
|
* </ul>
|
|
|
|
* {@hide}
|
|
|
|
*/
|
|
|
|
public class VpnManager {
|
|
|
|
// Action for broadcasting a connectivity state.
|
|
|
|
private static final String ACTION_VPN_CONNECTIVITY = "vpn.connectivity";
|
|
|
|
/** Key to the profile name of a connectivity broadcast event. */
|
|
|
|
public static final String BROADCAST_PROFILE_NAME = "profile_name";
|
|
|
|
/** Key to the connectivity state of a connectivity broadcast event. */
|
|
|
|
public static final String BROADCAST_CONNECTION_STATE = "connection_state";
|
2009-07-11 14:23:30 +00:00
|
|
|
/** Key to the error code of a connectivity broadcast event. */
|
|
|
|
public static final String BROADCAST_ERROR_CODE = "err";
|
|
|
|
/** Error code to indicate an error from authentication. */
|
2009-08-27 10:45:21 +00:00
|
|
|
public static final int VPN_ERROR_AUTH = 51;
|
2009-07-11 14:23:30 +00:00
|
|
|
/** Error code to indicate the connection attempt failed. */
|
2009-08-27 10:45:21 +00:00
|
|
|
public static final int VPN_ERROR_CONNECTION_FAILED = 101;
|
2009-07-11 14:23:30 +00:00
|
|
|
/** Error code to indicate the server is not known. */
|
2009-08-27 10:45:21 +00:00
|
|
|
public static final int VPN_ERROR_UNKNOWN_SERVER = 102;
|
2009-07-30 06:02:48 +00:00
|
|
|
/** Error code to indicate an error from challenge response. */
|
2009-08-27 10:45:21 +00:00
|
|
|
public static final int VPN_ERROR_CHALLENGE = 5;
|
2009-07-30 06:02:48 +00:00
|
|
|
/** Error code to indicate an error of remote server hanging up. */
|
2009-08-27 10:45:21 +00:00
|
|
|
public static final int VPN_ERROR_REMOTE_HUNG_UP = 7;
|
|
|
|
/** Error code to indicate an error of remote PPP server hanging up. */
|
|
|
|
public static final int VPN_ERROR_REMOTE_PPP_HUNG_UP = 48;
|
|
|
|
/** Error code to indicate a PPP negotiation error. */
|
|
|
|
public static final int VPN_ERROR_PPP_NEGOTIATION_FAILED = 42;
|
2009-08-03 08:22:24 +00:00
|
|
|
/** Error code to indicate an error of losing connectivity. */
|
2009-08-27 10:45:21 +00:00
|
|
|
public static final int VPN_ERROR_CONNECTION_LOST = 103;
|
|
|
|
/** Largest error code used by VPN. */
|
|
|
|
public static final int VPN_ERROR_LARGEST = 200;
|
|
|
|
/** Error code to indicate a successful connection. */
|
|
|
|
public static final int VPN_ERROR_NO_ERROR = 0;
|
2009-06-03 15:56:34 +00:00
|
|
|
|
2009-11-25 04:49:24 +00:00
|
|
|
public static final String PROFILES_PATH = "/misc/vpn/profiles";
|
2009-06-03 15:56:34 +00:00
|
|
|
|
|
|
|
private static final String PACKAGE_PREFIX =
|
|
|
|
VpnManager.class.getPackage().getName() + ".";
|
|
|
|
|
2009-06-08 05:27:11 +00:00
|
|
|
// Action to start VPN service
|
|
|
|
private static final String ACTION_VPN_SERVICE = PACKAGE_PREFIX + "SERVICE";
|
2009-06-03 15:56:34 +00:00
|
|
|
|
|
|
|
// Action to start VPN settings
|
2009-07-11 14:23:30 +00:00
|
|
|
private static final String ACTION_VPN_SETTINGS =
|
|
|
|
PACKAGE_PREFIX + "SETTINGS";
|
2009-06-03 15:56:34 +00:00
|
|
|
|
2009-11-18 18:14:20 +00:00
|
|
|
public static final String TAG = VpnManager.class.getSimpleName();
|
|
|
|
|
|
|
|
// TODO(oam): Test VPN when EFS is enabled (will do later)...
|
|
|
|
public static String getProfilePath() {
|
|
|
|
// This call will return the correct path if Encrypted FS is enabled or not.
|
|
|
|
return Environment.getSecureDataDirectory().getPath() + PROFILES_PATH;
|
|
|
|
}
|
2009-06-03 15:56:34 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns all supported VPN types.
|
|
|
|
*/
|
|
|
|
public static VpnType[] getSupportedVpnTypes() {
|
|
|
|
return VpnType.values();
|
|
|
|
}
|
|
|
|
|
|
|
|
private Context mContext;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a manager object with the specified context.
|
|
|
|
*/
|
|
|
|
public VpnManager(Context c) {
|
|
|
|
mContext = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a VPN profile of the specified type.
|
|
|
|
*
|
|
|
|
* @param type the VPN type
|
|
|
|
* @return the profile object
|
|
|
|
*/
|
|
|
|
public VpnProfile createVpnProfile(VpnType type) {
|
|
|
|
return createVpnProfile(type, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a VPN profile of the specified type.
|
|
|
|
*
|
|
|
|
* @param type the VPN type
|
|
|
|
* @param customized true if the profile is custom made
|
|
|
|
* @return the profile object
|
|
|
|
*/
|
|
|
|
public VpnProfile createVpnProfile(VpnType type, boolean customized) {
|
|
|
|
try {
|
|
|
|
VpnProfile p = (VpnProfile) type.getProfileClass().newInstance();
|
|
|
|
p.setCustomized(customized);
|
|
|
|
return p;
|
|
|
|
} catch (InstantiationException e) {
|
|
|
|
return null;
|
|
|
|
} catch (IllegalAccessException e) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2009-06-08 05:27:11 +00:00
|
|
|
* Starts the VPN service to establish VPN connection.
|
2009-06-03 15:56:34 +00:00
|
|
|
*/
|
2009-06-08 05:27:11 +00:00
|
|
|
public void startVpnService() {
|
|
|
|
mContext.startService(new Intent(ACTION_VPN_SERVICE));
|
2009-06-03 15:56:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2009-06-08 05:27:11 +00:00
|
|
|
* Stops the VPN service.
|
2009-06-03 15:56:34 +00:00
|
|
|
*/
|
2009-06-08 05:27:11 +00:00
|
|
|
public void stopVpnService() {
|
|
|
|
mContext.stopService(new Intent(ACTION_VPN_SERVICE));
|
2009-06-03 15:56:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2009-06-08 05:27:11 +00:00
|
|
|
* Binds the specified ServiceConnection with the VPN service.
|
2009-06-03 15:56:34 +00:00
|
|
|
*/
|
2009-06-08 05:27:11 +00:00
|
|
|
public boolean bindVpnService(ServiceConnection c) {
|
|
|
|
if (!mContext.bindService(new Intent(ACTION_VPN_SERVICE), c, 0)) {
|
|
|
|
Log.w(TAG, "failed to connect to VPN service");
|
2009-06-03 15:56:34 +00:00
|
|
|
return false;
|
|
|
|
} else {
|
2009-06-08 05:27:11 +00:00
|
|
|
Log.d(TAG, "succeeded to connect to VPN service");
|
2009-06-03 15:56:34 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Broadcasts the connectivity state of the specified profile. */
|
|
|
|
public void broadcastConnectivity(String profileName, VpnState s) {
|
2009-07-11 14:23:30 +00:00
|
|
|
broadcastConnectivity(profileName, s, VPN_ERROR_NO_ERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Broadcasts the connectivity state with an error code. */
|
|
|
|
public void broadcastConnectivity(String profileName, VpnState s,
|
|
|
|
int error) {
|
2009-06-03 15:56:34 +00:00
|
|
|
Intent intent = new Intent(ACTION_VPN_CONNECTIVITY);
|
|
|
|
intent.putExtra(BROADCAST_PROFILE_NAME, profileName);
|
|
|
|
intent.putExtra(BROADCAST_CONNECTION_STATE, s);
|
2009-07-11 14:23:30 +00:00
|
|
|
if (error != VPN_ERROR_NO_ERROR) {
|
|
|
|
intent.putExtra(BROADCAST_ERROR_CODE, error);
|
|
|
|
}
|
2009-06-03 15:56:34 +00:00
|
|
|
mContext.sendBroadcast(intent);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void registerConnectivityReceiver(BroadcastReceiver r) {
|
|
|
|
IntentFilter filter = new IntentFilter();
|
|
|
|
filter.addAction(VpnManager.ACTION_VPN_CONNECTIVITY);
|
|
|
|
mContext.registerReceiver(r, filter);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void unregisterConnectivityReceiver(BroadcastReceiver r) {
|
|
|
|
mContext.unregisterReceiver(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Starts the VPN settings activity. */
|
|
|
|
public void startSettingsActivity() {
|
|
|
|
Intent intent = new Intent(ACTION_VPN_SETTINGS);
|
|
|
|
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
|
|
|
|
mContext.startActivity(intent);
|
|
|
|
}
|
|
|
|
|
2009-06-08 05:27:11 +00:00
|
|
|
/** Creates an intent to start the VPN settings activity. */
|
|
|
|
public Intent createSettingsActivityIntent() {
|
|
|
|
Intent intent = new Intent(ACTION_VPN_SETTINGS);
|
2009-06-03 15:56:34 +00:00
|
|
|
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
|
2009-06-08 05:27:11 +00:00
|
|
|
return intent;
|
2009-06-03 15:56:34 +00:00
|
|
|
}
|
|
|
|
}
|