mirror of https://github.com/PurpleI2P/i2pd.git
hypnosis-i2p
9 years ago
14 changed files with 521 additions and 69 deletions
@ -0,0 +1,16 @@ |
|||||||
|
<menu |
||||||
|
xmlns:android="http://schemas.android.com/apk/res/android" |
||||||
|
xmlns:app="http://schemas.android.com/apk/res-auto" |
||||||
|
xmlns:tools="http://schemas.android.com/tools" |
||||||
|
tools:context=".I2PD"> |
||||||
|
<item |
||||||
|
android:id="@+id/action_graceful_quit" |
||||||
|
android:title="@string/action_graceful_quit" |
||||||
|
android:orderInCategory="98" |
||||||
|
/> |
||||||
|
<item |
||||||
|
android:id="@+id/action_quit" |
||||||
|
android:title="@string/action_quit" |
||||||
|
android:orderInCategory="99" |
||||||
|
/> |
||||||
|
</menu> |
@ -1,4 +1,9 @@ |
|||||||
<?xml version="1.0" encoding="utf-8"?> |
<?xml version="1.0" encoding="utf-8"?> |
||||||
<resources> |
<resources> |
||||||
<string name="app_name">i2pd</string> |
<string name="app_name">i2pd</string> |
||||||
|
<string name="i2pd_started">i2pd started</string> |
||||||
|
<string name="action_quit">Quit</string> |
||||||
|
<string name="action_graceful_quit">Graceful Quit</string> |
||||||
|
<string name="graceful_quit_is_already_in_progress">Graceful quit is already in progress</string> |
||||||
|
<string name="graceful_quit_is_in_progress">Graceful quit is in progress</string> |
||||||
</resources> |
</resources> |
||||||
|
@ -0,0 +1,88 @@ |
|||||||
|
package org.purplei2p.i2pd; |
||||||
|
|
||||||
|
import android.app.Notification; |
||||||
|
import android.app.PendingIntent; |
||||||
|
import android.app.Service; |
||||||
|
import android.content.Intent; |
||||||
|
import android.os.Binder; |
||||||
|
import android.os.IBinder; |
||||||
|
import android.support.v4.app.NotificationCompat; |
||||||
|
import android.util.Log; |
||||||
|
|
||||||
|
public class ForegroundService extends Service { |
||||||
|
// private NotificationManager mNM;
|
||||||
|
|
||||||
|
// Unique Identification Number for the Notification.
|
||||||
|
// We use it on Notification start, and to cancel it.
|
||||||
|
private int NOTIFICATION = R.string.i2pd_started; |
||||||
|
|
||||||
|
/** |
||||||
|
* Class for clients to access. Because we know this service always |
||||||
|
* runs in the same process as its clients, we don't need to deal with |
||||||
|
* IPC. |
||||||
|
*/ |
||||||
|
public class LocalBinder extends Binder { |
||||||
|
ForegroundService getService() { |
||||||
|
return ForegroundService.this; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onCreate() { |
||||||
|
// mNM = (NotificationManager)getSystemService(NOTIFICATION_SERVICE);
|
||||||
|
|
||||||
|
// Display a notification about us starting. We put an icon in the status bar.
|
||||||
|
showNotification(); |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public int onStartCommand(Intent intent, int flags, int startId) { |
||||||
|
Log.i("ForegroundService", "Received start id " + startId + ": " + intent); |
||||||
|
return START_NOT_STICKY; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public void onDestroy() { |
||||||
|
// Cancel the persistent notification.
|
||||||
|
//mNM.cancel(NOTIFICATION);
|
||||||
|
stopForeground(true); |
||||||
|
|
||||||
|
// Tell the user we stopped.
|
||||||
|
//Toast.makeText(this, R.string.local_service_stopped, Toast.LENGTH_SHORT).show();
|
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public IBinder onBind(Intent intent) { |
||||||
|
return mBinder; |
||||||
|
} |
||||||
|
|
||||||
|
// This is the object that receives interactions from clients. See
|
||||||
|
// RemoteService for a more complete example.
|
||||||
|
private final IBinder mBinder = new LocalBinder(); |
||||||
|
|
||||||
|
/** |
||||||
|
* Show a notification while this service is running. |
||||||
|
*/ |
||||||
|
private void showNotification() { |
||||||
|
// In this sample, we'll use the same text for the ticker and the expanded notification
|
||||||
|
CharSequence text = getText(R.string.i2pd_started); |
||||||
|
|
||||||
|
// The PendingIntent to launch our activity if the user selects this notification
|
||||||
|
PendingIntent contentIntent = PendingIntent.getActivity(this, 0, |
||||||
|
new Intent(this, I2PD.class), 0); |
||||||
|
|
||||||
|
// Set the info for the views that show in the notification panel.
|
||||||
|
Notification notification = new NotificationCompat.Builder(this) |
||||||
|
.setSmallIcon(R.drawable.itoopie_notification_icon) // the status icon
|
||||||
|
.setTicker(text) // the status text
|
||||||
|
.setWhen(System.currentTimeMillis()) // the time stamp
|
||||||
|
.setContentTitle(getText(R.string.app_name)) // the label of the entry
|
||||||
|
.setContentText(text) // the contents of the entry
|
||||||
|
.setContentIntent(contentIntent) // The intent to send when the entry is clicked
|
||||||
|
.build(); |
||||||
|
|
||||||
|
// Send the notification.
|
||||||
|
//mNM.notify(NOTIFICATION, notification);
|
||||||
|
startForeground(NOTIFICATION, notification); |
||||||
|
} |
||||||
|
} |
@ -1,20 +1,272 @@ |
|||||||
package org.purplei2p.i2pd; |
package org.purplei2p.i2pd; |
||||||
|
|
||||||
|
import java.io.PrintWriter; |
||||||
|
import java.io.StringWriter; |
||||||
|
import java.util.Timer; |
||||||
|
import java.util.TimerTask; |
||||||
|
|
||||||
|
import android.annotation.SuppressLint; |
||||||
import android.app.Activity; |
import android.app.Activity; |
||||||
import android.widget.TextView; |
import android.content.ComponentName; |
||||||
|
import android.content.Context; |
||||||
|
import android.content.Intent; |
||||||
|
import android.content.ServiceConnection; |
||||||
|
import android.os.Build; |
||||||
import android.os.Bundle; |
import android.os.Bundle; |
||||||
|
import android.os.IBinder; |
||||||
|
import android.util.Log; |
||||||
|
import android.view.Menu; |
||||||
|
import android.view.MenuItem; |
||||||
|
import android.widget.TextView; |
||||||
|
import android.widget.Toast; |
||||||
|
|
||||||
public class I2PD extends Activity { |
public class I2PD extends Activity { |
||||||
|
private static final String TAG = "i2pd"; |
||||||
|
|
||||||
|
private static Throwable loadLibsThrowable; |
||||||
|
static { |
||||||
|
try { |
||||||
|
I2PD_JNI.loadLibraries(); |
||||||
|
} catch (Throwable tr) { |
||||||
|
loadLibsThrowable = tr; |
||||||
|
} |
||||||
|
} |
||||||
|
private String daemonStartResult="N/A"; |
||||||
|
private boolean destroyed=false; |
||||||
|
private TextView textView; |
||||||
|
|
||||||
@Override |
@Override |
||||||
public void onCreate(Bundle savedInstanceState) { |
public void onCreate(Bundle savedInstanceState) { |
||||||
super.onCreate(savedInstanceState); |
super.onCreate(savedInstanceState); |
||||||
|
destroyed=false; |
||||||
|
|
||||||
TextView tv = new TextView(this); |
//set the app be foreground (do not unload when RAM needed)
|
||||||
tv.setText( "libi2pd.so was compiled with ABI " + getABICompiledWith()); |
doBindService(); |
||||||
setContentView(tv); |
|
||||||
|
textView = new TextView(this); |
||||||
|
setContentView(textView); |
||||||
|
if (loadLibsThrowable != null) { |
||||||
|
textView.setText(throwableToString(loadLibsThrowable)+"\r\n"); |
||||||
|
return; |
||||||
|
} |
||||||
|
try { |
||||||
|
textView.setText( |
||||||
|
"libi2pd.so was compiled with ABI " + getABICompiledWith() + "\r\n"+ |
||||||
|
"Starting daemon... "); |
||||||
|
new Thread(new Runnable(){ |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
try { |
||||||
|
doStartDaemon(); |
||||||
|
} catch (final Throwable tr) { |
||||||
|
appendThrowable(tr); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
},"i2pdDaemonStarting").start(); |
||||||
|
} catch (Throwable tr) { |
||||||
|
textView.setText(textView.getText().toString()+throwableToString(tr)); |
||||||
|
} |
||||||
} |
} |
||||||
|
|
||||||
public String getABICompiledWith() { |
@Override |
||||||
|
protected void onDestroy() { |
||||||
|
super.onDestroy(); |
||||||
|
localDestroy(); |
||||||
|
} |
||||||
|
|
||||||
|
private synchronized void localDestroy() { |
||||||
|
if(destroyed)return; |
||||||
|
destroyed=true; |
||||||
|
if(gracefulQuitTimer!=null) { |
||||||
|
gracefulQuitTimer.cancel(); |
||||||
|
gracefulQuitTimer = null; |
||||||
|
} |
||||||
|
try{ |
||||||
|
doUnbindService(); |
||||||
|
}catch(Throwable tr){ |
||||||
|
Log.e(TAG, "", tr); |
||||||
|
} |
||||||
|
if("ok".equals(daemonStartResult)) { |
||||||
|
try { |
||||||
|
I2PD_JNI.stopDaemon(); |
||||||
|
} catch (Throwable tr) { |
||||||
|
Log.e(TAG, "error", tr); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private CharSequence throwableToString(Throwable tr) { |
||||||
|
StringWriter sw = new StringWriter(8192); |
||||||
|
PrintWriter pw = new PrintWriter(sw); |
||||||
|
tr.printStackTrace(pw); |
||||||
|
pw.close(); |
||||||
|
return sw.toString(); |
||||||
|
} |
||||||
|
|
||||||
|
public String getABICompiledWith() { |
||||||
return I2PD_JNI.getABICompiledWith(); |
return I2PD_JNI.getABICompiledWith(); |
||||||
} |
} |
||||||
|
|
||||||
|
private synchronized void doStartDaemon() { |
||||||
|
if(destroyed)return; |
||||||
|
daemonStartResult = I2PD_JNI.startDaemon(); |
||||||
|
runOnUiThread(new Runnable(){ |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
synchronized (I2PD.this) { |
||||||
|
if(destroyed)return; |
||||||
|
textView.setText( |
||||||
|
textView.getText().toString()+ |
||||||
|
"start result: "+daemonStartResult+"\r\n" |
||||||
|
); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
private void appendThrowable(final Throwable tr) { |
||||||
|
runOnUiThread(new Runnable(){ |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
synchronized (I2PD.this) { |
||||||
|
if(destroyed)return; |
||||||
|
textView.setText(textView.getText().toString()+throwableToString(tr)+"\r\n"); |
||||||
|
} |
||||||
|
} |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
// private LocalService mBoundService;
|
||||||
|
|
||||||
|
private ServiceConnection mConnection = new ServiceConnection() { |
||||||
|
public void onServiceConnected(ComponentName className, IBinder service) { |
||||||
|
// This is called when the connection with the service has been
|
||||||
|
// established, giving us the service object we can use to
|
||||||
|
// interact with the service. Because we have bound to a explicit
|
||||||
|
// service that we know is running in our own process, we can
|
||||||
|
// cast its IBinder to a concrete class and directly access it.
|
||||||
|
// mBoundService = ((LocalService.LocalBinder)service).getService();
|
||||||
|
|
||||||
|
// Tell the user about this for our demo.
|
||||||
|
// Toast.makeText(Binding.this, R.string.local_service_connected,
|
||||||
|
// Toast.LENGTH_SHORT).show();
|
||||||
|
} |
||||||
|
|
||||||
|
public void onServiceDisconnected(ComponentName className) { |
||||||
|
// This is called when the connection with the service has been
|
||||||
|
// unexpectedly disconnected -- that is, its process crashed.
|
||||||
|
// Because it is running in our same process, we should never
|
||||||
|
// see this happen.
|
||||||
|
// mBoundService = null;
|
||||||
|
// Toast.makeText(Binding.this, R.string.local_service_disconnected,
|
||||||
|
// Toast.LENGTH_SHORT).show();
|
||||||
|
} |
||||||
|
}; |
||||||
|
|
||||||
|
|
||||||
|
private boolean mIsBound; |
||||||
|
|
||||||
|
private void doBindService() { |
||||||
|
// Establish a connection with the service. We use an explicit
|
||||||
|
// class name because we want a specific service implementation that
|
||||||
|
// we know will be running in our own process (and thus won't be
|
||||||
|
// supporting component replacement by other applications).
|
||||||
|
bindService(new Intent(this, |
||||||
|
ForegroundService.class), mConnection, Context.BIND_AUTO_CREATE); |
||||||
|
mIsBound = true; |
||||||
|
} |
||||||
|
|
||||||
|
private void doUnbindService() { |
||||||
|
if (mIsBound) { |
||||||
|
// Detach our existing connection.
|
||||||
|
unbindService(mConnection); |
||||||
|
mIsBound = false; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
@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_quit: |
||||||
|
quit(); |
||||||
|
return true; |
||||||
|
case R.id.action_graceful_quit: |
||||||
|
gracefulQuit(); |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
return super.onOptionsItemSelected(item); |
||||||
|
} |
||||||
|
|
||||||
|
@SuppressLint("NewApi") |
||||||
|
private void quit() { |
||||||
|
try { |
||||||
|
localDestroy(); |
||||||
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { |
||||||
|
finishAndRemoveTask(); |
||||||
|
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) { |
||||||
|
finishAffinity(); |
||||||
|
} else { |
||||||
|
//moveTaskToBack(true);
|
||||||
|
finish(); |
||||||
|
} |
||||||
|
}catch (Throwable tr) { |
||||||
|
Log.e(TAG, "", tr); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
private Timer gracefulQuitTimer; |
||||||
|
private synchronized void gracefulQuit() { |
||||||
|
if(gracefulQuitTimer!=null){ |
||||||
|
Toast.makeText(this, R.string.graceful_quit_is_already_in_progress, |
||||||
|
Toast.LENGTH_SHORT).show(); |
||||||
|
return; |
||||||
|
} |
||||||
|
Toast.makeText(this, R.string.graceful_quit_is_in_progress, |
||||||
|
Toast.LENGTH_SHORT).show(); |
||||||
|
new Thread(new Runnable(){ |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
try{ |
||||||
|
synchronized (I2PD.this) { |
||||||
|
if("ok".equals(daemonStartResult)) { |
||||||
|
I2PD_JNI.stopAcceptingTunnels(); |
||||||
|
gracefulQuitTimer = new Timer(true); |
||||||
|
gracefulQuitTimer.schedule(new TimerTask(){ |
||||||
|
|
||||||
|
@Override |
||||||
|
public void run() { |
||||||
|
quit(); |
||||||
|
} |
||||||
|
|
||||||
|
}, 10*60*1000/*millis*/); |
||||||
|
}else{ |
||||||
|
quit(); |
||||||
|
} |
||||||
|
} |
||||||
|
} catch(Throwable tr) { |
||||||
|
Log.e(TAG,"",tr); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
},"gracQuitInit").start(); |
||||||
|
} |
||||||
} |
} |
||||||
|
@ -0,0 +1,30 @@ |
|||||||
|
package org.purplei2p.i2pd; |
||||||
|
|
||||||
|
import android.util.Log; |
||||||
|
import android.content.BroadcastReceiver; |
||||||
|
import android.content.Context; |
||||||
|
import android.content.Intent; |
||||||
|
import android.net.ConnectivityManager; |
||||||
|
import android.net.NetworkInfo; |
||||||
|
|
||||||
|
public class NetworkStateChangeReceiver extends BroadcastReceiver { |
||||||
|
|
||||||
|
private static final String TAG = "i2pd"; |
||||||
|
|
||||||
|
//api level 1
|
||||||
|
@Override |
||||||
|
public void onReceive(final Context context, final Intent intent) { |
||||||
|
Log.d(TAG,"Network state change"); |
||||||
|
try { |
||||||
|
ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE); |
||||||
|
NetworkInfo activeNetworkInfo = cm.getActiveNetworkInfo(); |
||||||
|
boolean isConnected = activeNetworkInfo!=null && activeNetworkInfo.isConnected(); |
||||||
|
// https://developer.android.com/training/monitoring-device-state/connectivity-monitoring.html?hl=ru
|
||||||
|
// boolean isWiFi = activeNetworkInfo!=null && (activeNetworkInfo.getType() == ConnectivityManager.TYPE_WIFI);
|
||||||
|
|
||||||
|
I2PD_JNI.onNetworkStateChanged(isConnected); |
||||||
|
} catch (Throwable tr) { |
||||||
|
Log.d(TAG,"",tr); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue