summaryrefslogtreecommitdiff
path: root/main/src
diff options
context:
space:
mode:
Diffstat (limited to 'main/src')
-rw-r--r--main/src/ui/java/de/blinkt/openvpn/activities/ConfigConverter.java846
-rw-r--r--main/src/ui/java/de/blinkt/openvpn/activities/ConfigConverter.kt797
2 files changed, 797 insertions, 846 deletions
diff --git a/main/src/ui/java/de/blinkt/openvpn/activities/ConfigConverter.java b/main/src/ui/java/de/blinkt/openvpn/activities/ConfigConverter.java
deleted file mode 100644
index cd6133be..00000000
--- a/main/src/ui/java/de/blinkt/openvpn/activities/ConfigConverter.java
+++ /dev/null
@@ -1,846 +0,0 @@
-
-/*
- * Copyright (c) 2012-2016 Arne Schwabe
- * Distributed under the GNU GPL v2 with additional terms. For full terms see the file doc/LICENSE.txt
- */
-
-package de.blinkt.openvpn.activities;
-
-import android.Manifest;
-import android.annotation.TargetApi;
-import android.app.Activity;
-import android.app.AlertDialog;
-import android.app.AlertDialog.Builder;
-import android.content.ActivityNotFoundException;
-import android.content.Intent;
-import android.content.pm.PackageManager;
-import android.database.Cursor;
-import android.net.Uri;
-import android.os.AsyncTask;
-import android.os.Build;
-import android.os.Bundle;
-import android.os.Environment;
-import android.provider.OpenableColumns;
-import android.security.KeyChain;
-import android.security.KeyChainAliasCallback;
-import androidx.annotation.NonNull;
-import android.text.TextUtils;
-import android.util.Base64;
-import android.util.Pair;
-import android.view.Menu;
-import android.view.MenuInflater;
-import android.view.MenuItem;
-import android.view.View;
-import android.view.ViewGroup;
-import android.widget.CheckBox;
-import android.widget.EditText;
-import android.widget.ImageButton;
-import android.widget.LinearLayout;
-import android.widget.ProgressBar;
-import android.widget.TextView;
-import android.widget.Toast;
-
-import java.io.File;
-import java.io.FileInputStream;
-import java.io.IOException;
-import java.io.InputStream;
-import java.io.InputStreamReader;
-import java.io.UnsupportedEncodingException;
-import java.net.URLDecoder;
-import java.util.HashMap;
-import java.util.HashSet;
-import java.util.List;
-import java.util.Map;
-import java.util.Vector;
-
-import de.blinkt.openvpn.R;
-import de.blinkt.openvpn.VpnProfile;
-import de.blinkt.openvpn.core.ConfigParser;
-import de.blinkt.openvpn.core.ConfigParser.ConfigParseError;
-import de.blinkt.openvpn.core.ProfileManager;
-import de.blinkt.openvpn.fragments.Utils;
-import de.blinkt.openvpn.views.FileSelectLayout;
-
-import static de.blinkt.openvpn.views.FileSelectLayout.FileSelectCallback;
-
-public class ConfigConverter extends BaseActivity implements FileSelectCallback, View.OnClickListener {
-
- public static final String IMPORT_PROFILE = "de.blinkt.openvpn.IMPORT_PROFILE";
- private static final int RESULT_INSTALLPKCS12 = 7;
- private static final int CHOOSE_FILE_OFFSET = 1000;
- public static final String VPNPROFILE = "vpnProfile";
- private static final int PERMISSION_REQUEST_EMBED_FILES = 37231;
- private static final int PERMISSION_REQUEST_READ_URL = PERMISSION_REQUEST_EMBED_FILES + 1;
-
- private VpnProfile mResult;
-
- private transient List<String> mPathsegments;
-
- private String mAliasName = null;
-
-
- private Map<Utils.FileType, FileSelectLayout> fileSelectMap = new HashMap<>();
- private String mEmbeddedPwFile;
- private Vector<String> mLogEntries = new Vector<>();
- private Uri mSourceUri;
- private EditText mProfilename;
- private AsyncTask<Void, Void, Integer> mImportTask;
- private LinearLayout mLogLayout;
- private TextView mProfilenameLabel;
-
- @Override
- public void onClick(View v) {
- if (v.getId() == R.id.fab_save)
- userActionSaveProfile();
- if (v.getId() == R.id.permssion_hint && Build.VERSION.SDK_INT == Build.VERSION_CODES.M)
- doRequestSDCardPermission(PERMISSION_REQUEST_EMBED_FILES);
-
- }
-
- @TargetApi(Build.VERSION_CODES.M)
- private void doRequestSDCardPermission(int requestCode) {
- requestPermissions(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, requestCode);
- }
-
- @Override
- public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
- // Permission declined, do nothing
- if (grantResults.length == 0 || grantResults[0] == PackageManager.PERMISSION_DENIED)
- return;
-
- // Reset file select dialogs
- findViewById(R.id.files_missing_hint).setVisibility(View.GONE);
- findViewById(R.id.permssion_hint).setVisibility(View.GONE);
- LinearLayout fileroot = (LinearLayout) findViewById(R.id.config_convert_root);
- for (int i = 0; i < fileroot.getChildCount(); ) {
- if (fileroot.getChildAt(i) instanceof FileSelectLayout)
- fileroot.removeViewAt(i);
- else
- i++;
- }
-
- if (requestCode == PERMISSION_REQUEST_EMBED_FILES)
- embedFiles(null);
-
- else if (requestCode == PERMISSION_REQUEST_READ_URL) {
- if (mSourceUri != null)
- doImportUri(mSourceUri);
- }
- }
-
- @Override
- public boolean onOptionsItemSelected(MenuItem item) {
- if (item.getItemId() == R.id.cancel) {
- setResult(Activity.RESULT_CANCELED);
- finish();
- } else if (item.getItemId() == R.id.ok) {
- return userActionSaveProfile();
- }
-
- return super.onOptionsItemSelected(item);
-
- }
-
- private boolean userActionSaveProfile() {
- if (mResult == null) {
- log(R.string.import_config_error);
- Toast.makeText(this, R.string.import_config_error, Toast.LENGTH_LONG).show();
- return true;
- }
-
- mResult.mName = mProfilename.getText().toString();
- ProfileManager vpl = ProfileManager.getInstance(this);
- if (vpl.getProfileByName(mResult.mName) != null) {
- mProfilename.setError(getString(R.string.duplicate_profile_name));
- return true;
- }
-
- Intent in = installPKCS12();
-
- if (in != null)
- startActivityForResult(in, RESULT_INSTALLPKCS12);
- else
- saveProfile();
-
- return true;
- }
-
- @Override
- protected void onSaveInstanceState(@NonNull Bundle outState) {
- super.onSaveInstanceState(outState);
- if (mResult != null)
- outState.putSerializable(VPNPROFILE, mResult);
- outState.putString("mAliasName", mAliasName);
-
-
- String[] logentries = mLogEntries.toArray(new String[mLogEntries.size()]);
-
- outState.putStringArray("logentries", logentries);
-
- int[] fileselects = new int[fileSelectMap.size()];
- int k = 0;
- for (Utils.FileType key : fileSelectMap.keySet()) {
- fileselects[k] = key.getValue();
- k++;
- }
- outState.putIntArray("fileselects", fileselects);
- outState.putString("pwfile", mEmbeddedPwFile);
- outState.putParcelable("mSourceUri", mSourceUri);
- }
-
- @Override
- protected void onActivityResult(int requestCode, int resultCode, Intent result) {
- if (requestCode == RESULT_INSTALLPKCS12 && resultCode == Activity.RESULT_OK) {
- showCertDialog();
- }
-
- if (resultCode == Activity.RESULT_OK && requestCode >= CHOOSE_FILE_OFFSET) {
- Utils.FileType type = Utils.FileType.getFileTypeByValue(requestCode - CHOOSE_FILE_OFFSET);
-
-
- FileSelectLayout fs = fileSelectMap.get(type);
- fs.parseResponse(result, this);
-
- String data = fs.getData();
-
- switch (type) {
- case USERPW_FILE:
- mEmbeddedPwFile = data;
- break;
- case PKCS12:
- mResult.mPKCS12Filename = data;
- break;
- case TLS_AUTH_FILE:
- mResult.mTLSAuthFilename = data;
- break;
- case CA_CERTIFICATE:
- mResult.mCaFilename = data;
- break;
- case CLIENT_CERTIFICATE:
- mResult.mClientCertFilename = data;
- break;
- case KEYFILE:
- mResult.mClientKeyFilename = data;
- break;
- case CRL_FILE:
- mResult.mCrlFilename = data;
- break;
- default:
- throw new RuntimeException("Type is wrong somehow?");
- }
- }
-
- super.onActivityResult(requestCode, resultCode, result);
- }
-
- private void saveProfile() {
- Intent result = new Intent();
- ProfileManager vpl = ProfileManager.getInstance(this);
-
- if (!TextUtils.isEmpty(mEmbeddedPwFile))
- ConfigParser.useEmbbedUserAuth(mResult, mEmbeddedPwFile);
-
- vpl.addProfile(mResult);
- vpl.saveProfile(this, mResult);
- vpl.saveProfileList(this);
- result.putExtra(VpnProfile.EXTRA_PROFILEUUID, mResult.getUUID().toString());
- setResult(Activity.RESULT_OK, result);
- finish();
- }
-
- public void showCertDialog() {
- try {
- //noinspection WrongConstant
- KeyChain.choosePrivateKeyAlias(this,
- new KeyChainAliasCallback() {
-
- public void alias(String alias) {
- // Credential alias selected. Remember the alias selection for future use.
- mResult.mAlias = alias;
- saveProfile();
- }
-
-
- },
- new String[]{"RSA", "EC"}, // List of acceptable key types. null for any
- null, // issuer, null for any
- mResult.mServerName, // host name of server requesting the cert, null if unavailable
- -1, // port of server requesting the cert, -1 if unavailable
- mAliasName); // alias to preselect, null if unavailable
- } catch (ActivityNotFoundException anf) {
- Builder ab = new AlertDialog.Builder(this);
- ab.setTitle(R.string.broken_image_cert_title);
- ab.setMessage(R.string.broken_image_cert);
- ab.setPositiveButton(android.R.string.ok, null);
- ab.show();
- }
- }
-
-
- private Intent installPKCS12() {
-
- if (!((CheckBox) findViewById(R.id.importpkcs12)).isChecked()) {
- setAuthTypeToEmbeddedPKCS12();
- return null;
-
- }
- String pkcs12datastr = mResult.mPKCS12Filename;
- if (VpnProfile.isEmbedded(pkcs12datastr)) {
- Intent inkeyIntent = KeyChain.createInstallIntent();
-
- pkcs12datastr = VpnProfile.getEmbeddedContent(pkcs12datastr);
-
-
- byte[] pkcs12data = Base64.decode(pkcs12datastr, Base64.DEFAULT);
-
-
- inkeyIntent.putExtra(KeyChain.EXTRA_PKCS12, pkcs12data);
-
- if (mAliasName.equals(""))
- mAliasName = null;
-
- if (mAliasName != null) {
- inkeyIntent.putExtra(KeyChain.EXTRA_NAME, mAliasName);
- }
- return inkeyIntent;
-
- }
- return null;
- }
-
-
- private void setAuthTypeToEmbeddedPKCS12() {
- if (VpnProfile.isEmbedded(mResult.mPKCS12Filename)) {
- if (mResult.mAuthenticationType == VpnProfile.TYPE_USERPASS_KEYSTORE)
- mResult.mAuthenticationType = VpnProfile.TYPE_USERPASS_PKCS12;
-
- if (mResult.mAuthenticationType == VpnProfile.TYPE_KEYSTORE)
- mResult.mAuthenticationType = VpnProfile.TYPE_PKCS12;
-
- }
- }
-
-
- private String getUniqueProfileName(String possibleName) {
-
- int i = 0;
-
- ProfileManager vpl = ProfileManager.getInstance(this);
-
- String newname = possibleName;
-
- // Default to
- if (mResult.mName != null && !ConfigParser.CONVERTED_PROFILE.equals(mResult.mName))
- newname = mResult.mName;
-
- while (newname == null || vpl.getProfileByName(newname) != null) {
- i++;
- if (i == 1)
- newname = getString(R.string.converted_profile);
- else
- newname = getString(R.string.converted_profile_i, i);
- }
-
- return newname;
- }
-
- @Override
- public boolean onCreateOptionsMenu(Menu menu) {
- MenuInflater inflater = getMenuInflater();
- inflater.inflate(R.menu.import_menu, menu);
- return true;
- }
-
- private String embedFile(String filename, Utils.FileType type, boolean onlyFindFileAndNullonNotFound) {
- if (filename == null)
- return null;
-
- // Already embedded, nothing to do
- if (VpnProfile.isEmbedded(filename))
- return filename;
-
- File possibleFile = findFile(filename, type);
- if (possibleFile == null)
- if (onlyFindFileAndNullonNotFound)
- return null;
- else
- return filename;
- else if (onlyFindFileAndNullonNotFound)
- return possibleFile.getAbsolutePath();
- else
- return readFileContent(possibleFile, type == Utils.FileType.PKCS12);
-
- }
-
-
- private Pair<Integer, String> getFileDialogInfo(Utils.FileType type) {
- int titleRes = 0;
- String value = null;
- switch (type) {
- case KEYFILE:
- titleRes = R.string.client_key_title;
- if (mResult != null)
- value = mResult.mClientKeyFilename;
- break;
- case CLIENT_CERTIFICATE:
- titleRes = R.string.client_certificate_title;
- if (mResult != null)
- value = mResult.mClientCertFilename;
- break;
- case CA_CERTIFICATE:
- titleRes = R.string.ca_title;
- if (mResult != null)
- value = mResult.mCaFilename;
- break;
- case TLS_AUTH_FILE:
- titleRes = R.string.tls_auth_file;
- if (mResult != null)
- value = mResult.mTLSAuthFilename;
- break;
- case PKCS12:
- titleRes = R.string.client_pkcs12_title;
- if (mResult != null)
- value = mResult.mPKCS12Filename;
- break;
-
- case USERPW_FILE:
- titleRes = R.string.userpw_file;
- value = mEmbeddedPwFile;
- break;
-
- case CRL_FILE:
- titleRes = R.string.crl_file;
- value = mResult.mCrlFilename;
- break;
- }
-
- return Pair.create(titleRes, value);
-
- }
-
- private File findFile(String filename, Utils.FileType fileType) {
- File foundfile = findFileRaw(filename);
-
- if (foundfile == null && filename != null && !filename.equals("")) {
- log(R.string.import_could_not_open, filename);
- }
- fileSelectMap.put(fileType, null);
-
- return foundfile;
- }
-
- private void addMissingFileDialogs()
- {
- for (Map.Entry<Utils.FileType, FileSelectLayout> item: fileSelectMap.entrySet()) {
- if (item.getValue()==null)
- addFileSelectDialog(item.getKey());
- }
- }
-
- private void addFileSelectDialog(Utils.FileType type) {
-
- Pair<Integer, String> fileDialogInfo = getFileDialogInfo(type);
-
- boolean isCert = type == Utils.FileType.CA_CERTIFICATE || type == Utils.FileType.CLIENT_CERTIFICATE;
- FileSelectLayout fl = new FileSelectLayout(this, getString(fileDialogInfo.first), isCert, false);
- fileSelectMap.put(type, fl);
- fl.setLayoutParams(new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT));
-
- ((LinearLayout) findViewById(R.id.config_convert_root)).addView(fl, 2);
- findViewById(R.id.files_missing_hint).setVisibility(View.VISIBLE);
- if (Build.VERSION.SDK_INT == Build.VERSION_CODES.M)
- checkPermission();
-
- fl.setData(fileDialogInfo.second, this);
- int i = getFileLayoutOffset(type);
- fl.setCaller(this, i, type);
-
- }
-
- @TargetApi(Build.VERSION_CODES.M)
- private void checkPermission() {
- if (checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
- findViewById(R.id.permssion_hint).setVisibility(View.VISIBLE);
- findViewById(R.id.permssion_hint).setOnClickListener(this);
- }
- }
-
- private int getFileLayoutOffset(Utils.FileType type) {
- return CHOOSE_FILE_OFFSET + type.getValue();
- }
-
-
- private File findFileRaw(String filename) {
- if (filename == null || filename.equals(""))
- return null;
-
- // Try diffent path relative to /mnt/sdcard
- File sdcard = Environment.getExternalStorageDirectory();
- File root = new File("/");
-
- HashSet<File> dirlist = new HashSet<>();
-
- for (int i = mPathsegments.size() - 1; i >= 0; i--) {
- String path = "";
- for (int j = 0; j <= i; j++) {
- path += "/" + mPathsegments.get(j);
- }
- // Do a little hackish dance for the Android File Importer
- // /document/primary:ovpn/openvpn-imt.conf
-
-
- if (path.indexOf(':') != -1 && path.lastIndexOf('/') > path.indexOf(':')) {
- String possibleDir = path.substring(path.indexOf(':') + 1, path.length());
- // Unquote chars in the path
- try {
- possibleDir = URLDecoder.decode(possibleDir, "UTF-8");
- } catch (UnsupportedEncodingException ignored) {}
-
- possibleDir = possibleDir.substring(0, possibleDir.lastIndexOf('/'));
-
-
-
-
- dirlist.add(new File(sdcard, possibleDir));
-
- }
- dirlist.add(new File(path));
-
-
- }
- dirlist.add(sdcard);
- dirlist.add(root);
-
-
- String[] fileparts = filename.split("/");
- for (File rootdir : dirlist) {
- String suffix = "";
- for (int i = fileparts.length - 1; i >= 0; i--) {
- if (i == fileparts.length - 1)
- suffix = fileparts[i];
- else
- suffix = fileparts[i] + "/" + suffix;
-
- File possibleFile = new File(rootdir, suffix);
- if (possibleFile.canRead())
- return possibleFile;
-
- }
- }
- return null;
- }
-
- String readFileContent(File possibleFile, boolean base64encode) {
- byte[] filedata;
- try {
- filedata = readBytesFromFile(possibleFile);
- } catch (IOException e) {
- log(e.getLocalizedMessage());
- return null;
- }
-
- String data;
- if (base64encode) {
- data = Base64.encodeToString(filedata, Base64.DEFAULT);
- } else {
- data = new String(filedata);
-
- }
-
- return VpnProfile.DISPLAYNAME_TAG + possibleFile.getName() + VpnProfile.INLINE_TAG + data;
-
- }
-
-
- private byte[] readBytesFromFile(File file) throws IOException {
- InputStream input = new FileInputStream(file);
-
- long len = file.length();
- if (len > VpnProfile.MAX_EMBED_FILE_SIZE)
- throw new IOException("File size of file to import too large.");
-
- // Create the byte array to hold the data
- byte[] bytes = new byte[(int) len];
-
- // Read in the bytes
- int offset = 0;
- int bytesRead;
- while (offset < bytes.length
- && (bytesRead = input.read(bytes, offset, bytes.length - offset)) >= 0) {
- offset += bytesRead;
- }
-
- input.close();
- return bytes;
- }
-
- void embedFiles(ConfigParser cp) {
- // This where I would like to have a c++ style
- // void embedFile(std::string & option)
-
- if (mResult.mPKCS12Filename != null) {
- File pkcs12file = findFileRaw(mResult.mPKCS12Filename);
- if (pkcs12file != null) {
- mAliasName = pkcs12file.getName().replace(".p12", "");
- } else {
- mAliasName = "Imported PKCS12";
- }
- }
-
-
- mResult.mCaFilename = embedFile(mResult.mCaFilename, Utils.FileType.CA_CERTIFICATE, false);
- mResult.mClientCertFilename = embedFile(mResult.mClientCertFilename, Utils.FileType.CLIENT_CERTIFICATE, false);
- mResult.mClientKeyFilename = embedFile(mResult.mClientKeyFilename, Utils.FileType.KEYFILE, false);
- mResult.mTLSAuthFilename = embedFile(mResult.mTLSAuthFilename, Utils.FileType.TLS_AUTH_FILE, false);
- mResult.mPKCS12Filename = embedFile(mResult.mPKCS12Filename, Utils.FileType.PKCS12, false);
- mResult.mCrlFilename = embedFile(mResult.mCrlFilename, Utils.FileType.CRL_FILE, true);
- if (cp != null) {
- mEmbeddedPwFile = cp.getAuthUserPassFile();
- mEmbeddedPwFile = embedFile(cp.getAuthUserPassFile(), Utils.FileType.USERPW_FILE, false);
- }
-
- }
-
- private void updateFileSelectDialogs() {
- for (Map.Entry<Utils.FileType, FileSelectLayout> fl : fileSelectMap.entrySet()) {
- fl.getValue().setData(getFileDialogInfo(fl.getKey()).second, this);
- }
- }
-
-
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.config_converter);
-
- ImageButton fab_button = (ImageButton) findViewById(R.id.fab_save);
- if (fab_button != null) {
- fab_button.setOnClickListener(this);
- findViewById(R.id.fab_footerspace).setVisibility(View.VISIBLE);
- }
-
- mLogLayout = (LinearLayout) findViewById(R.id.config_convert_root);
-
-
- mProfilename = (EditText) findViewById(R.id.profilename);
- mProfilenameLabel = (TextView) findViewById(R.id.profilename_label);
-
- if (savedInstanceState != null && savedInstanceState.containsKey(VPNPROFILE)) {
- mResult = (VpnProfile) savedInstanceState.getSerializable(VPNPROFILE);
- mAliasName = savedInstanceState.getString("mAliasName");
- mEmbeddedPwFile = savedInstanceState.getString("pwfile");
- mSourceUri = savedInstanceState.getParcelable("mSourceUri");
- mProfilename.setText(mResult.mName);
-
- if (savedInstanceState.containsKey("logentries")) {
- //noinspection ConstantConditions
- for (String logItem : savedInstanceState.getStringArray("logentries"))
- log(logItem);
- }
- if (savedInstanceState.containsKey("fileselects")) {
- //noinspection ConstantConditions
- for (int k : savedInstanceState.getIntArray("fileselects")) {
- addFileSelectDialog(Utils.FileType.getFileTypeByValue(k));
- }
- }
- return;
- }
-
-
- final android.content.Intent intent = getIntent();
-
- if (intent != null) {
- doImportIntent(intent);
-
- // We parsed the intent, relay on saved instance for restoring
- setIntent(null);
- }
-
-
- }
-
- private void doImportIntent(Intent intent) {
- final Uri data = intent.getData();
- if (data != null) {
- mSourceUri = data;
- doImportUri(data);
- }
- }
-
- private void doImportUri(Uri data) {
- //log(R.string.import_experimental);
- log(R.string.importing_config, data.toString());
- String possibleName = null;
- if ((data.getScheme() != null && data.getScheme().equals("file")) ||
- (data.getLastPathSegment() != null &&
- (data.getLastPathSegment().endsWith(".ovpn") ||
- data.getLastPathSegment().endsWith(".conf")))
- ) {
- possibleName = data.getLastPathSegment();
- if (possibleName.lastIndexOf('/') != -1)
- possibleName = possibleName.substring(possibleName.lastIndexOf('/') + 1);
-
- }
-
- mPathsegments = data.getPathSegments();
-
- Cursor cursor = getContentResolver().query(data, null, null, null, null);
-
- try {
-
- if (cursor != null && cursor.moveToFirst()) {
- int columnIndex = cursor.getColumnIndex(OpenableColumns.DISPLAY_NAME);
-
- if (columnIndex != -1) {
- String displayName = cursor.getString(columnIndex);
- if (displayName != null)
- possibleName = displayName;
- }
- columnIndex = cursor.getColumnIndex("mime_type");
- if (columnIndex != -1) {
- log("Mime type: " + cursor.getString(columnIndex));
- }
- }
- } finally {
- if (cursor != null)
- cursor.close();
- }
- if (possibleName != null) {
- possibleName = possibleName.replace(".ovpn", "");
- possibleName = possibleName.replace(".conf", "");
- }
-
- startImportTask(data, possibleName);
-
-
- }
-
- private void startImportTask(final Uri data, final String possibleName) {
- mImportTask = new AsyncTask<Void, Void, Integer>() {
- private ProgressBar mProgress;
-
- @Override
- protected void onPreExecute() {
- mProgress = new ProgressBar(ConfigConverter.this);
- addViewToLog(mProgress);
- }
-
- @Override
- protected Integer doInBackground(Void... params) {
- try {
- InputStream is = getContentResolver().openInputStream(data);
-
- doImport(is);
- is.close();
- if (mResult==null)
- return -3;
- } catch (IOException| SecurityException se)
-
- {
- log(R.string.import_content_resolve_error + ":" + se.getLocalizedMessage());
- if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
- checkMarschmallowFileImportError(data);
- return -2;
- }
-
- return 0;
- }
-
- @Override
- protected void onPostExecute(Integer errorCode) {
- mLogLayout.removeView(mProgress);
- addMissingFileDialogs();
- updateFileSelectDialogs();
-
- if (errorCode == 0) {
- displayWarnings();
- mResult.mName = getUniqueProfileName(possibleName);
- mProfilename.setVisibility(View.VISIBLE);
- mProfilenameLabel.setVisibility(View.VISIBLE);
- mProfilename.setText(mResult.getName());
-
- log(R.string.import_done);
- }
- }
- }.execute();
- }
-
-
- @TargetApi(Build.VERSION_CODES.M)
- private void checkMarschmallowFileImportError(Uri data) {
- // Permission already granted, not the source of the error
- if (checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED)
- return;
-
- // We got a file:/// URL and have no permission to read it. Technically an error of the calling app since
- // it makes an assumption about other apps being able to read the url but well ...
- if (data != null && "file".equals(data.getScheme()))
- doRequestSDCardPermission(PERMISSION_REQUEST_READ_URL);
-
- }
-
-
- @Override
- protected void onStart() {
- super.onStart();
- }
-
- private void log(final String logmessage) {
- runOnUiThread(new Runnable() {
- @Override
- public void run() {
- TextView tv = new TextView(ConfigConverter.this);
- mLogEntries.add(logmessage);
- tv.setText(logmessage);
-
- addViewToLog(tv);
- }
- });
- }
-
- private void addViewToLog(View view) {
- mLogLayout.addView(view, mLogLayout.getChildCount() - 1);
- }
-
- private void doImport(InputStream is) {
- ConfigParser cp = new ConfigParser();
- try {
- InputStreamReader isr = new InputStreamReader(is);
-
- cp.parseConfig(isr);
- mResult = cp.convertProfile();
- embedFiles(cp);
- return;
-
- } catch (IOException | ConfigParseError e) {
- log(R.string.error_reading_config_file);
- log(e.getLocalizedMessage());
- }
- mResult = null;
-
- }
-
- private void displayWarnings() {
- if (mResult.mUseCustomConfig) {
- log(R.string.import_warning_custom_options);
- String copt = mResult.mCustomConfigOptions;
- if (copt.startsWith("#")) {
- int until = copt.indexOf('\n');
- copt = copt.substring(until + 1);
- }
-
- log(copt);
- }
-
- if (mResult.mAuthenticationType == VpnProfile.TYPE_KEYSTORE ||
- mResult.mAuthenticationType == VpnProfile.TYPE_USERPASS_KEYSTORE) {
- findViewById(R.id.importpkcs12).setVisibility(View.VISIBLE);
- }
-
- }
-
- private void log(int ressourceId, Object... formatArgs) {
- log(getString(ressourceId, formatArgs));
- }
-
-}
diff --git a/main/src/ui/java/de/blinkt/openvpn/activities/ConfigConverter.kt b/main/src/ui/java/de/blinkt/openvpn/activities/ConfigConverter.kt
new file mode 100644
index 00000000..672ffae6
--- /dev/null
+++ b/main/src/ui/java/de/blinkt/openvpn/activities/ConfigConverter.kt
@@ -0,0 +1,797 @@
+/*
+ * Copyright (c) 2012-2016 Arne Schwabe
+ * Distributed under the GNU GPL v2 with additional terms. For full terms see the file doc/LICENSE.txt
+ */
+
+package de.blinkt.openvpn.activities
+
+import android.Manifest
+import android.annotation.TargetApi
+import android.app.Activity
+import android.app.AlertDialog
+import android.content.ActivityNotFoundException
+import android.content.Intent
+import android.content.pm.PackageManager
+import android.net.Uri
+import android.os.AsyncTask
+import android.os.Build
+import android.os.Bundle
+import android.os.Environment
+import android.provider.OpenableColumns
+import android.security.KeyChain
+import android.text.TextUtils
+import android.util.Base64
+import android.util.Pair
+import android.view.Menu
+import android.view.MenuItem
+import android.view.View
+import android.view.ViewGroup
+import android.widget.*
+import de.blinkt.openvpn.R
+import de.blinkt.openvpn.VpnProfile
+import de.blinkt.openvpn.core.ConfigParser
+import de.blinkt.openvpn.core.ConfigParser.ConfigParseError
+import de.blinkt.openvpn.core.ProfileManager
+import de.blinkt.openvpn.fragments.Utils
+import de.blinkt.openvpn.views.FileSelectLayout
+import de.blinkt.openvpn.views.FileSelectLayout.FileSelectCallback
+import java.io.*
+import java.net.URLDecoder
+import java.util.*
+
+class ConfigConverter : BaseActivity(), FileSelectCallback, View.OnClickListener {
+
+ private var mResult: VpnProfile? = null
+
+ @Transient
+ private var mPathsegments: List<String>? = null
+
+ private var mAliasName: String? = null
+
+
+ private val fileSelectMap = HashMap<Utils.FileType, FileSelectLayout?>()
+ private var mEmbeddedPwFile: String? = null
+ private val mLogEntries = Vector<String>()
+ private var mSourceUri: Uri? = null
+ private lateinit var mProfilename: EditText
+ private var mImportTask: AsyncTask<Void, Void, Int>? = null
+ private lateinit var mLogLayout: LinearLayout
+ private lateinit var mProfilenameLabel: TextView
+
+ override fun onClick(v: View) {
+ if (v.id == R.id.fab_save)
+ userActionSaveProfile()
+ if (v.id == R.id.permssion_hint && Build.VERSION.SDK_INT == Build.VERSION_CODES.M)
+ doRequestSDCardPermission(PERMISSION_REQUEST_EMBED_FILES)
+
+ }
+
+ @TargetApi(Build.VERSION_CODES.M)
+ private fun doRequestSDCardPermission(requestCode: Int) {
+ requestPermissions(arrayOf(Manifest.permission.READ_EXTERNAL_STORAGE), requestCode)
+ }
+
+ override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<String>, grantResults: IntArray) {
+ // Permission declined, do nothing
+ if (grantResults.size == 0 || grantResults[0] == PackageManager.PERMISSION_DENIED)
+ return
+
+ // Reset file select dialogs
+ findViewById<View>(R.id.files_missing_hint).visibility = View.GONE
+ findViewById<View>(R.id.permssion_hint).visibility = View.GONE
+ val fileroot = findViewById<View>(R.id.config_convert_root) as LinearLayout
+ var i = 0
+ while (i < fileroot.childCount) {
+ if (fileroot.getChildAt(i) is FileSelectLayout)
+ fileroot.removeViewAt(i)
+ else
+ i++
+ }
+
+ if (requestCode == PERMISSION_REQUEST_EMBED_FILES)
+ embedFiles(null)
+ else if (requestCode == PERMISSION_REQUEST_READ_URL) {
+ if (mSourceUri != null)
+ doImportUri(mSourceUri!!)
+ }
+ }
+
+ override fun onOptionsItemSelected(item: MenuItem): Boolean {
+ if (item.itemId == R.id.cancel) {
+ setResult(Activity.RESULT_CANCELED)
+ finish()
+ } else if (item.itemId == R.id.ok) {
+ return userActionSaveProfile()
+ }
+
+ return super.onOptionsItemSelected(item)
+
+ }
+
+ private fun userActionSaveProfile(): Boolean {
+ if (mResult == null) {
+ log(R.string.import_config_error)
+ Toast.makeText(this, R.string.import_config_error, Toast.LENGTH_LONG).show()
+ return true
+ }
+
+ mResult!!.mName = mProfilename.text.toString()
+ val vpl = ProfileManager.getInstance(this)
+ if (vpl.getProfileByName(mResult!!.mName) != null) {
+ mProfilename.error = getString(R.string.duplicate_profile_name)
+ return true
+ }
+
+ val `in` = installPKCS12()
+
+ if (`in` != null)
+ startActivityForResult(`in`, RESULT_INSTALLPKCS12)
+ else
+ saveProfile()
+
+ return true
+ }
+
+ override fun onSaveInstanceState(outState: Bundle) {
+ super.onSaveInstanceState(outState)
+ if (mResult != null)
+ outState.putSerializable(VPNPROFILE, mResult)
+ outState.putString("mAliasName", mAliasName)
+
+
+ val logentries = mLogEntries.toTypedArray()
+
+ outState.putStringArray("logentries", logentries)
+
+ val fileselects = IntArray(fileSelectMap.size)
+ var k = 0
+ for (key in fileSelectMap.keys) {
+ fileselects[k] = key.value
+ k++
+ }
+ outState.putIntArray("fileselects", fileselects)
+ outState.putString("pwfile", mEmbeddedPwFile)
+ outState.putParcelable("mSourceUri", mSourceUri)
+ }
+
+ override fun onActivityResult(requestCode: Int, resultCode: Int, result: Intent) {
+ if (requestCode == RESULT_INSTALLPKCS12 && resultCode == Activity.RESULT_OK) {
+ showCertDialog()
+ }
+
+ if (resultCode == Activity.RESULT_OK && requestCode >= CHOOSE_FILE_OFFSET) {
+ val type = Utils.FileType.getFileTypeByValue(requestCode - CHOOSE_FILE_OFFSET)
+
+
+ val fs = fileSelectMap[type]
+ fs!!.parseResponse(result, this)
+
+ val data = fs.data
+
+ when (type) {
+ Utils.FileType.USERPW_FILE -> mEmbeddedPwFile = data
+ Utils.FileType.PKCS12 -> mResult!!.mPKCS12Filename = data
+ Utils.FileType.TLS_AUTH_FILE -> mResult!!.mTLSAuthFilename = data
+ Utils.FileType.CA_CERTIFICATE -> mResult!!.mCaFilename = data
+ Utils.FileType.CLIENT_CERTIFICATE -> mResult!!.mClientCertFilename = data
+ Utils.FileType.KEYFILE -> mResult!!.mClientKeyFilename = data
+ Utils.FileType.CRL_FILE -> mResult!!.mCrlFilename = data
+ else -> throw RuntimeException("Type is wrong somehow?")
+ }
+ }
+
+ super.onActivityResult(requestCode, resultCode, result)
+ }
+
+ private fun saveProfile() {
+ val result = Intent()
+ val vpl = ProfileManager.getInstance(this)
+
+ if (!TextUtils.isEmpty(mEmbeddedPwFile))
+ ConfigParser.useEmbbedUserAuth(mResult, mEmbeddedPwFile)
+
+ vpl.addProfile(mResult)
+ vpl.saveProfile(this, mResult)
+ vpl.saveProfileList(this)
+ result.putExtra(VpnProfile.EXTRA_PROFILEUUID, mResult!!.uuid.toString())
+ setResult(Activity.RESULT_OK, result)
+ finish()
+ }
+
+ fun showCertDialog() {
+ try {
+
+ KeyChain.choosePrivateKeyAlias(this,
+ { alias ->
+ // Credential alias selected. Remember the alias selection for future use.
+ mResult!!.mAlias = alias
+ saveProfile()
+ },
+ arrayOf("RSA", "EC"), null, // issuer, null for any
+ mResult!!.mServerName, // host name of server requesting the cert, null if unavailable
+ -1, // port of server requesting the cert, -1 if unavailable
+ mAliasName)// List of acceptable key types. null for any
+ // alias to preselect, null if unavailable
+ } catch (anf: ActivityNotFoundException) {
+ val ab = AlertDialog.Builder(this)
+ ab.setTitle(R.string.broken_image_cert_title)
+ ab.setMessage(R.string.broken_image_cert)
+ ab.setPositiveButton(android.R.string.ok, null)
+ ab.show()
+ }
+
+ }
+
+
+ private fun installPKCS12(): Intent? {
+
+ if (!(findViewById<View>(R.id.importpkcs12) as CheckBox).isChecked) {
+ setAuthTypeToEmbeddedPKCS12()
+ return null
+
+ }
+ var pkcs12datastr = mResult!!.mPKCS12Filename
+ if (VpnProfile.isEmbedded(pkcs12datastr)) {
+ val inkeyIntent = KeyChain.createInstallIntent()
+
+ pkcs12datastr = VpnProfile.getEmbeddedContent(pkcs12datastr)
+
+
+ val pkcs12data = Base64.decode(pkcs12datastr, Base64.DEFAULT)
+
+
+ inkeyIntent.putExtra(KeyChain.EXTRA_PKCS12, pkcs12data)
+
+ if (mAliasName == "")
+ mAliasName = null
+
+ if (mAliasName != null) {
+ inkeyIntent.putExtra(KeyChain.EXTRA_NAME, mAliasName)
+ }
+ return inkeyIntent
+
+ }
+ return null
+ }
+
+
+ private fun setAuthTypeToEmbeddedPKCS12() {
+ if (VpnProfile.isEmbedded(mResult!!.mPKCS12Filename)) {
+ if (mResult!!.mAuthenticationType == VpnProfile.TYPE_USERPASS_KEYSTORE)
+ mResult!!.mAuthenticationType = VpnProfile.TYPE_USERPASS_PKCS12
+
+ if (mResult!!.mAuthenticationType == VpnProfile.TYPE_KEYSTORE)
+ mResult!!.mAuthenticationType = VpnProfile.TYPE_PKCS12
+
+ }
+ }
+
+
+ private fun getUniqueProfileName(possibleName: String?): String {
+
+ var i = 0
+
+ val vpl = ProfileManager.getInstance(this)
+
+ var newname = possibleName
+
+ // Default to
+ if (mResult!!.mName != null && ConfigParser.CONVERTED_PROFILE != mResult!!.mName)
+ newname = mResult!!.mName
+
+ while (newname == null || vpl.getProfileByName(newname) != null) {
+ i++
+ if (i == 1)
+ newname = getString(R.string.converted_profile)
+ else
+ newname = getString(R.string.converted_profile_i, i)
+ }
+
+ return newname
+ }
+
+ override fun onCreateOptionsMenu(menu: Menu): Boolean {
+ val inflater = menuInflater
+ inflater.inflate(R.menu.import_menu, menu)
+ return true
+ }
+
+ private fun embedFile(filename: String?, type: Utils.FileType, onlyFindFileAndNullonNotFound: Boolean): String? {
+ if (filename == null)
+ return null
+
+ // Already embedded, nothing to do
+ if (VpnProfile.isEmbedded(filename))
+ return filename
+
+ val possibleFile = findFile(filename, type)
+ return if (possibleFile == null)
+ if (onlyFindFileAndNullonNotFound)
+ null
+ else
+ filename
+ else if (onlyFindFileAndNullonNotFound)
+ possibleFile.absolutePath
+ else
+ readFileContent(possibleFile, type == Utils.FileType.PKCS12)
+
+ }
+
+
+ private fun getFileDialogInfo(type: Utils.FileType): Pair<Int, String> {
+ var titleRes = 0
+ var value: String? = null
+ when (type) {
+ Utils.FileType.KEYFILE -> {
+ titleRes = R.string.client_key_title
+ if (mResult != null)
+ value = mResult!!.mClientKeyFilename
+ }
+ Utils.FileType.CLIENT_CERTIFICATE -> {
+ titleRes = R.string.client_certificate_title
+ if (mResult != null)
+ value = mResult!!.mClientCertFilename
+ }
+ Utils.FileType.CA_CERTIFICATE -> {
+ titleRes = R.string.ca_title
+ if (mResult != null)
+ value = mResult!!.mCaFilename
+ }
+ Utils.FileType.TLS_AUTH_FILE -> {
+ titleRes = R.string.tls_auth_file
+ if (mResult != null)
+ value = mResult!!.mTLSAuthFilename
+ }
+ Utils.FileType.PKCS12 -> {
+ titleRes = R.string.client_pkcs12_title
+ if (mResult != null)
+ value = mResult!!.mPKCS12Filename
+ }
+
+ Utils.FileType.USERPW_FILE -> {
+ titleRes = R.string.userpw_file
+ value = mEmbeddedPwFile
+ }
+
+ Utils.FileType.CRL_FILE -> {
+ titleRes = R.string.crl_file
+ value = mResult!!.mCrlFilename
+ }
+ Utils.FileType.OVPN_CONFIG -> TODO()
+ }
+
+ return Pair.create(titleRes, value)
+
+ }
+
+ private fun findFile(filename: String?, fileType: Utils.FileType): File? {
+ val foundfile = findFileRaw(filename)
+
+ if (foundfile == null && filename != null && filename != "") {
+ log(R.string.import_could_not_open, filename)
+ }
+ fileSelectMap[fileType] = null
+
+ return foundfile
+ }
+
+ private fun addMissingFileDialogs() {
+ for ((key, value) in fileSelectMap) {
+ if (value == null)
+ addFileSelectDialog(key)
+ }
+ }
+
+ private fun addFileSelectDialog(type: Utils.FileType?) {
+
+ val fileDialogInfo = getFileDialogInfo(type!!)
+
+ val isCert = type == Utils.FileType.CA_CERTIFICATE || type == Utils.FileType.CLIENT_CERTIFICATE
+ val fl = FileSelectLayout(this, getString(fileDialogInfo.first), isCert, false)
+ fileSelectMap[type] = fl
+ fl.layoutParams = LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT)
+
+ (findViewById<View>(R.id.config_convert_root) as LinearLayout).addView(fl, 2)
+ findViewById<View>(R.id.files_missing_hint).visibility = View.VISIBLE
+ if (Build.VERSION.SDK_INT == Build.VERSION_CODES.M)
+ checkPermission()
+
+ fl.setData(fileDialogInfo.second, this)
+ val i = getFileLayoutOffset(type)
+ fl.setCaller(this, i, type)
+
+ }
+
+ @TargetApi(Build.VERSION_CODES.M)
+ private fun checkPermission() {
+ if (checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
+ findViewById<View>(R.id.permssion_hint).visibility = View.VISIBLE
+ findViewById<View>(R.id.permssion_hint).setOnClickListener(this)
+ }
+ }
+
+ private fun getFileLayoutOffset(type: Utils.FileType): Int {
+ return CHOOSE_FILE_OFFSET + type.value
+ }
+
+
+ private fun findFileRaw(filename: String?): File? {
+ if (filename == null || filename == "")
+ return null
+
+ // Try diffent path relative to /mnt/sdcard
+ val sdcard = Environment.getExternalStorageDirectory()
+ val root = File("/")
+
+ val dirlist = HashSet<File>()
+
+ for (i in mPathsegments!!.indices.reversed()) {
+ var path = ""
+ for (j in 0..i) {
+ path += "/" + mPathsegments!![j]
+ }
+ // Do a little hackish dance for the Android File Importer
+ // /document/primary:ovpn/openvpn-imt.conf
+
+
+ if (path.indexOf(':') != -1 && path.lastIndexOf('/') > path.indexOf(':')) {
+ var possibleDir = path.substring(path.indexOf(':') + 1, path.length)
+ // Unquote chars in the path
+ try {
+ possibleDir = URLDecoder.decode(possibleDir, "UTF-8")
+ } catch (ignored: UnsupportedEncodingException) {
+ }
+
+ possibleDir = possibleDir.substring(0, possibleDir.lastIndexOf('/'))
+
+
+
+
+ dirlist.add(File(sdcard, possibleDir))
+
+ }
+ dirlist.add(File(path))
+
+
+ }
+ dirlist.add(sdcard)
+ dirlist.add(root)
+
+
+ val fileparts = filename.split("/".toRegex()).dropLastWhile { it.isEmpty() }.toTypedArray()
+ for (rootdir in dirlist) {
+ var suffix = ""
+ for (i in fileparts.indices.reversed()) {
+ if (i == fileparts.size - 1)
+ suffix = fileparts[i]
+ else
+ suffix = fileparts[i] + "/" + suffix
+
+ val possibleFile = File(rootdir, suffix)
+ if (possibleFile.canRead())
+ return possibleFile
+
+ }
+ }
+ return null
+ }
+
+ internal fun readFileContent(possibleFile: File, base64encode: Boolean): String? {
+ val filedata: ByteArray
+ try {
+ filedata = readBytesFromFile(possibleFile)
+ } catch (e: IOException) {
+ log(e.localizedMessage)
+ return null
+ }
+
+ val data: String
+ if (base64encode) {
+ data = Base64.encodeToString(filedata, Base64.DEFAULT)
+ } else {
+ data = String(filedata)
+
+ }
+
+ return VpnProfile.DISPLAYNAME_TAG + possibleFile.name + VpnProfile.INLINE_TAG + data
+
+ }
+
+
+ @Throws(IOException::class)
+ private fun readBytesFromFile(file: File): ByteArray {
+ val input = FileInputStream(file)
+
+ val len = file.length()
+ if (len > VpnProfile.MAX_EMBED_FILE_SIZE)
+ throw IOException("File size of file to import too large.")
+
+ // Create the byte array to hold the data
+ val bytes = ByteArray(len.toInt())
+
+ // Read in the bytes
+ var offset = 0
+ var bytesRead:Int
+ do {
+ bytesRead = input.read(bytes, offset, bytes.size - offset)
+ offset += bytesRead
+ } while (offset < bytes.size && bytesRead >= 0)
+
+ input.close()
+ return bytes
+ }
+
+ internal fun embedFiles(cp: ConfigParser?) {
+ // This where I would like to have a c++ style
+ // void embedFile(std::string & option)
+
+ if (mResult!!.mPKCS12Filename != null) {
+ val pkcs12file = findFileRaw(mResult!!.mPKCS12Filename)
+ if (pkcs12file != null) {
+ mAliasName = pkcs12file.name.replace(".p12", "")
+ } else {
+ mAliasName = "Imported PKCS12"
+ }
+ }
+
+
+ mResult!!.mCaFilename = embedFile(mResult!!.mCaFilename, Utils.FileType.CA_CERTIFICATE, false)
+ mResult!!.mClientCertFilename = embedFile(mResult!!.mClientCertFilename, Utils.FileType.CLIENT_CERTIFICATE, false)
+ mResult!!.mClientKeyFilename = embedFile(mResult!!.mClientKeyFilename, Utils.FileType.KEYFILE, false)
+ mResult!!.mTLSAuthFilename = embedFile(mResult!!.mTLSAuthFilename, Utils.FileType.TLS_AUTH_FILE, false)
+ mResult!!.mPKCS12Filename = embedFile(mResult!!.mPKCS12Filename, Utils.FileType.PKCS12, false)
+ mResult!!.mCrlFilename = embedFile(mResult!!.mCrlFilename, Utils.FileType.CRL_FILE, true)
+ if (cp != null) {
+ mEmbeddedPwFile = cp.authUserPassFile
+ mEmbeddedPwFile = embedFile(cp.authUserPassFile, Utils.FileType.USERPW_FILE, false)
+ }
+
+ }
+
+ private fun updateFileSelectDialogs() {
+ for ((key, value) in fileSelectMap) {
+ value?.setData(getFileDialogInfo(key).second, this)
+ }
+ }
+
+
+ override fun onCreate(savedInstanceState: Bundle?) {
+ super.onCreate(savedInstanceState)
+ setContentView(R.layout.config_converter)
+
+ val fab_button = findViewById<ImageButton?>(R.id.fab_save)
+ if (fab_button != null) {
+ fab_button.setOnClickListener(this)
+ findViewById<View>(R.id.fab_footerspace).visibility = View.VISIBLE
+ }
+
+ mLogLayout = findViewById<View>(R.id.config_convert_root) as LinearLayout
+
+
+ mProfilename = findViewById<View>(R.id.profilename) as EditText
+ mProfilenameLabel = findViewById<View>(R.id.profilename_label) as TextView
+
+ if (savedInstanceState != null && savedInstanceState.containsKey(VPNPROFILE)) {
+ mResult = savedInstanceState.getSerializable(VPNPROFILE) as VpnProfile?
+ mAliasName = savedInstanceState.getString("mAliasName")
+ mEmbeddedPwFile = savedInstanceState.getString("pwfile")
+ mSourceUri = savedInstanceState.getParcelable("mSourceUri")
+ mProfilename.setText(mResult!!.mName)
+
+ if (savedInstanceState.containsKey("logentries")) {
+
+ for (logItem in savedInstanceState.getStringArray("logentries")!!)
+ log(logItem)
+ }
+ if (savedInstanceState.containsKey("fileselects")) {
+
+ for (k in savedInstanceState.getIntArray("fileselects")!!) {
+ addFileSelectDialog(Utils.FileType.getFileTypeByValue(k))
+ }
+ }
+ return
+ }
+
+
+ val intent = intent
+
+ if (intent != null) {
+ doImportIntent(intent)
+
+ // We parsed the intent, relay on saved instance for restoring
+ setIntent(null)
+ }
+
+
+ }
+
+ private fun doImportIntent(intent: Intent) {
+ val data = intent.data
+ if (data != null) {
+ mSourceUri = data
+ doImportUri(data)
+ }
+ }
+
+ private fun doImportUri(data: Uri) {
+ //log(R.string.import_experimental);
+ log(R.string.importing_config, data.toString())
+ var possibleName: String? = null
+ if (data.scheme != null && data.scheme == "file" || data.lastPathSegment != null && (data.lastPathSegment!!.endsWith(".ovpn") || data.lastPathSegment!!.endsWith(".conf"))) {
+ possibleName = data.lastPathSegment
+ if (possibleName!!.lastIndexOf('/') != -1)
+ possibleName = possibleName.substring(possibleName.lastIndexOf('/') + 1)
+
+ }
+
+ mPathsegments = data.pathSegments
+
+ val cursor = contentResolver.query(data, null, null, null, null)
+
+ try {
+
+ if (cursor != null && cursor.moveToFirst()) {
+ var columnIndex = cursor.getColumnIndex(OpenableColumns.DISPLAY_NAME)
+
+ if (columnIndex != -1) {
+ val displayName = cursor.getString(columnIndex)
+ if (displayName != null)
+ possibleName = displayName
+ }
+ columnIndex = cursor.getColumnIndex("mime_type")
+ if (columnIndex != -1) {
+ log("Mime type: " + cursor.getString(columnIndex))
+ }
+ }
+ } finally {
+ cursor?.close()
+ }
+ if (possibleName != null) {
+ possibleName = possibleName.replace(".ovpn", "")
+ possibleName = possibleName.replace(".conf", "")
+ }
+
+ startImportTask(data, possibleName)
+
+
+ }
+
+ private fun startImportTask(data: Uri, possibleName: String?) {
+ mImportTask = object : AsyncTask<Void, Void, Int>() {
+ private var mProgress: ProgressBar? = null
+
+ override fun onPreExecute() {
+ mProgress = ProgressBar(this@ConfigConverter)
+ addViewToLog(mProgress)
+ }
+
+ override fun doInBackground(vararg params: Void): Int? {
+ try {
+ val `is` = contentResolver.openInputStream(data)
+
+ doImport(`is`)
+ `is`!!.close()
+ if (mResult == null)
+ return -3
+ } catch (se: IOException) {
+ log(R.string.import_content_resolve_error.toString() + ":" + se.localizedMessage)
+ if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
+ checkMarschmallowFileImportError(data)
+ return -2
+ } catch (se: SecurityException) {
+ log(R.string.import_content_resolve_error.toString() + ":" + se.localizedMessage)
+ if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
+ checkMarschmallowFileImportError(data)
+ return -2
+ }
+
+ return 0
+ }
+
+ override fun onPostExecute(errorCode: Int?) {
+ mLogLayout.removeView(mProgress)
+ addMissingFileDialogs()
+ updateFileSelectDialogs()
+
+ if (errorCode == 0) {
+ displayWarnings()
+ mResult!!.mName = getUniqueProfileName(possibleName)
+ mProfilename.visibility = View.VISIBLE
+ mProfilenameLabel.visibility = View.VISIBLE
+ mProfilename!!.setText(mResult!!.name)
+
+ log(R.string.import_done)
+ }
+ }
+ }.execute()
+ }
+
+
+ @TargetApi(Build.VERSION_CODES.M)
+ private fun checkMarschmallowFileImportError(data: Uri?) {
+ // Permission already granted, not the source of the error
+ if (checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED)
+ return
+
+ // We got a file:/// URL and have no permission to read it. Technically an error of the calling app since
+ // it makes an assumption about other apps being able to read the url but well ...
+ if (data != null && "file" == data.scheme)
+ doRequestSDCardPermission(PERMISSION_REQUEST_READ_URL)
+
+ }
+
+
+ override fun onStart() {
+ super.onStart()
+ }
+
+ private fun log(logmessage: String?) {
+ runOnUiThread {
+ val tv = TextView(this@ConfigConverter)
+ mLogEntries.add(logmessage)
+ tv.text = logmessage
+
+ addViewToLog(tv)
+ }
+ }
+
+ private fun addViewToLog(view: View?) {
+ mLogLayout.addView(view, mLogLayout.childCount - 1)
+ }
+
+ private fun doImport(`is`: InputStream?) {
+ val cp = ConfigParser()
+ try {
+ val isr = InputStreamReader(`is`!!)
+
+ cp.parseConfig(isr)
+ mResult = cp.convertProfile()
+ embedFiles(cp)
+ return
+
+ } catch (e: IOException) {
+ log(R.string.error_reading_config_file)
+ log(e.localizedMessage)
+ } catch (e: ConfigParseError) {
+ log(R.string.error_reading_config_file)
+ log(e.localizedMessage)
+ }
+
+ mResult = null
+
+ }
+
+ private fun displayWarnings() {
+ if (mResult!!.mUseCustomConfig) {
+ log(R.string.import_warning_custom_options)
+ var copt = mResult!!.mCustomConfigOptions
+ if (copt.startsWith("#")) {
+ val until = copt.indexOf('\n')
+ copt = copt.substring(until + 1)
+ }
+
+ log(copt)
+ }
+
+ if (mResult!!.mAuthenticationType == VpnProfile.TYPE_KEYSTORE || mResult!!.mAuthenticationType == VpnProfile.TYPE_USERPASS_KEYSTORE) {
+ findViewById<View>(R.id.importpkcs12).visibility = View.VISIBLE
+ }
+
+ }
+
+ private fun log(ressourceId: Int, vararg formatArgs: Any) {
+ log(getString(ressourceId, *formatArgs))
+ }
+
+ companion object {
+
+ val IMPORT_PROFILE = "de.blinkt.openvpn.IMPORT_PROFILE"
+ val IMPORT_PROFILE_DATA = "de.blinkt.openvpn.IMPORT_PROFILE_DATA"
+ private val RESULT_INSTALLPKCS12 = 7
+ private val CHOOSE_FILE_OFFSET = 1000
+ val VPNPROFILE = "vpnProfile"
+ private val PERMISSION_REQUEST_EMBED_FILES = 37231
+ private val PERMISSION_REQUEST_READ_URL = PERMISSION_REQUEST_EMBED_FILES + 1
+ }
+
+}