package network.loki.lokinet; import java.io.File; import java.io.InputStream; import java.io.PrintWriter; import java.io.StringWriter; import java.net.URL; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import android.app.Activity; import android.content.Context; import android.content.ComponentName; import android.content.ServiceConnection; import android.Manifest; import android.net.VpnService; import android.os.AsyncTask; import android.content.Intent; import android.os.Bundle; import android.os.IBinder; import android.view.Menu; import android.view.MenuItem; import android.widget.TextView; import android.util.Log; public class LokiNetActivity extends Activity { private static final String TAG = "lokinet-activity"; private TextView textView; private static final String DefaultBootstrapURL = "https://seed.lokinet.org/lokinet.signed"; private AsyncBootstrap bootstrapper; public static final String LOG_TAG = "LokinetDaemon"; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); textView = new TextView(this); setContentView(textView); System.loadLibrary("lokinet-android"); } private static void writeFile(File out, InputStream instream) throws IOException { OutputStream outstream = new FileOutputStream(out); byte[] buffer = new byte[512]; int len; try { do { len = instream.read(buffer); if (len > 0) { outstream.write(buffer, 0, len); } } while (len != -1); } finally { outstream.close(); } } public void startLokinet() { if(bootstrapper != null) return; bootstrapper = new AsyncBootstrap(); bootstrapper.execute(DefaultBootstrapURL); } public void runLokinetService() { Intent intent = VpnService.prepare(getApplicationContext()); if (intent != null) { Log.d(LOG_TAG, "VpnService.prepare() returned an Intent, so launch that intent."); startActivityForResult(intent, 0); } else { Log.w(LOG_TAG, "VpnService.prepare() returned null, not running."); } } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { if (resultCode == RESULT_OK) { Log.d(LOG_TAG, "VpnService prepared intent RESULT_OK, launching LokinetDaemon Service"); startService(new Intent(LokiNetActivity.this, LokinetDaemon.class)); } else { Log.d(LOG_TAG, "VpnService prepared intent NOT RESULT_OK, shit."); } } @Override protected void onDestroy() { super.onDestroy(); textView = null; } public File getRootDir() { return getFilesDir(); } private class AsyncBootstrap extends AsyncTask { public String doInBackground(String ... urls) { try { File bootstrapFile = new File(getRootDir(), "bootstrap.signed"); URL bootstrapURL = new URL(urls[0]); InputStream instream = bootstrapURL.openStream(); writeFile(bootstrapFile, instream); instream.close(); return getString(R.string.bootstrap_ok); } catch(Exception thrown) { return getString(R.string.bootstrap_fail) + ": " + throwableToString(thrown); } } public void onPostExecute(String val) { textView.setText(val); if(val.equals(getString(R.string.bootstrap_ok))) runLokinetService(); bootstrapDone(); } } private void bootstrapDone() { bootstrapper = null; } private CharSequence throwableToString(Throwable tr) { StringWriter sw = new StringWriter(8192); PrintWriter pw = new PrintWriter(sw); tr.printStackTrace(pw); pw.close(); return sw.toString(); } @Override public boolean onCreateOptionsMenu(Menu menu) { // Inflate the menu; this adds items to the action bar if it is present. getMenuInflater().inflate(R.menu.options_main, menu); return true; } @Override public boolean onOptionsItemSelected(MenuItem item) { // Handle action bar item clicks here. The action bar will // automatically handle clicks on the Home/Up button, so long // as you specify a parent activity in AndroidManifest.xml. int id = item.getItemId(); switch(id){ case R.id.action_start: startLokinet(); return true; case R.id.action_stop: return true; } return super.onOptionsItemSelected(item); } }