wifi: can now play a full game

Made WifiDirectService into an actual service in order to better process
incoming packets. Now works for game messages from comms, and posts
notifications when app's in background. The latter requires using the
app context from XWApp since unlike the other transports this one
doesn't involve the OS invoking us with a Context.
This commit is contained in:
Eric House 2016-11-15 20:50:29 -08:00
parent 1c4ae5fcfd
commit 239beafcfa
10 changed files with 173 additions and 46 deletions

View file

@ -196,6 +196,9 @@
/> />
<service android:name="RelayService"/> <service android:name="RelayService"/>
<service android:name="BTService"/>
<service android:name="WifiDirectService"/>
<service android:name="SMSService"/>
<receiver android:name=".MountEventReceiver"> <receiver android:name=".MountEventReceiver">
<intent-filter> <intent-filter>
@ -217,8 +220,6 @@
</intent-filter> </intent-filter>
</receiver> </receiver>
<service android:name="BTService"/>
<receiver android:name="SMSReceiver" > <receiver android:name="SMSReceiver" >
<intent-filter> <intent-filter>
<action android:name="android.intent.action.DATA_SMS_RECEIVED" /> <action android:name="android.intent.action.DATA_SMS_RECEIVED" />
@ -229,8 +230,6 @@
</receiver> </receiver>
<service android:name="SMSService"/>
<receiver android:name="com.google.android.gcm.GCMBroadcastReceiver" <receiver android:name="com.google.android.gcm.GCMBroadcastReceiver"
android:permission="com.google.android.c2dm.permission.SEND" > android:permission="com.google.android.c2dm.permission.SEND" >
<intent-filter> <intent-filter>

View file

@ -43,6 +43,7 @@ public class BiDiSockWrap {
private boolean mRunThreads; private boolean mRunThreads;
private boolean mActive; private boolean mActive;
private InetAddress mAddress; private InetAddress mAddress;
private String mMacAddress; // of the remove device
private int mPort; private int mPort;
// For sockets that came from accept() on a ServerSocket // For sockets that came from accept() on a ServerSocket
@ -61,6 +62,16 @@ public class BiDiSockWrap {
mPort = port; mPort = port;
} }
public void setMacAddress( String addr )
{
mMacAddress = addr;
}
public String getMacAddress()
{
Assert.assertNotNull( mMacAddress ); return mMacAddress;
}
public BiDiSockWrap connect() public BiDiSockWrap connect()
{ {
mActive = true; mActive = true;

View file

@ -673,9 +673,9 @@ public class BoardDelegate extends DelegateBase
case RELAY_INVITE_RESULT: case RELAY_INVITE_RESULT:
missingMeans = InviteMeans.RELAY; missingMeans = InviteMeans.RELAY;
break; break;
case P2P_INVITE_RESULT: // case P2P_INVITE_RESULT:
missingMeans = InviteMeans.WIFIDIRECT; // missingMeans = InviteMeans.WIFIDIRECT;
break; // break;
} }
if ( null != missingMeans ) { if ( null != missingMeans ) {

View file

@ -2481,10 +2481,7 @@ public class DBUtils {
public static void appendLog( String tag, String msg ) public static void appendLog( String tag, String msg )
{ {
Context context = XWApp.getContext(); appendLog( XWApp.getContext(), msg );
if ( null != context ) {
appendLog( context, msg );
}
} }
private static void appendLog( Context context, String msg ) private static void appendLog( Context context, String msg )

View file

@ -290,7 +290,7 @@ public class DlgDelegate {
// These are stored in the INVITES table. Don't change order // These are stored in the INVITES table. Don't change order
// gratuitously // gratuitously
public static enum InviteMeans { public static enum InviteMeans {
SMS, EMAIL, NFC, BLUETOOTH, CLIPBOARD, RELAY, WIFIDIRECT, SMS, EMAIL, NFC, BLUETOOTH, CLIPBOARD, RELAY, // WIFIDIRECT,
}; };
void dlgButtonClicked( Action action, int button, Object[] params ); void dlgButtonClicked( Action action, int button, Object[] params );
void inviteChoiceMade( Action action, InviteMeans means, Object[] params ); void inviteChoiceMade( Action action, InviteMeans means, Object[] params );
@ -736,10 +736,10 @@ public class DlgDelegate {
items.add( getString( R.string.invite_choice_relay ) ); items.add( getString( R.string.invite_choice_relay ) );
means.add( DlgClickNotify.InviteMeans.RELAY ); means.add( DlgClickNotify.InviteMeans.RELAY );
} }
if ( WifiDirectService.supported() ) { // if ( WifiDirectService.supported() ) {
items.add( getString( R.string.invite_choice_p2p ) ); // items.add( getString( R.string.invite_choice_p2p ) );
means.add( DlgClickNotify.InviteMeans.WIFIDIRECT ); // means.add( DlgClickNotify.InviteMeans.WIFIDIRECT );
} // }
items.add( getString( R.string.slmenu_copy_sel ) ); items.add( getString( R.string.slmenu_copy_sel ) );
means.add( DlgClickNotify.InviteMeans.CLIPBOARD ); means.add( DlgClickNotify.InviteMeans.CLIPBOARD );

View file

@ -28,7 +28,7 @@ public enum RequestCode {
BT_INVITE_RESULT, BT_INVITE_RESULT,
SMS_INVITE_RESULT, SMS_INVITE_RESULT,
RELAY_INVITE_RESULT, RELAY_INVITE_RESULT,
P2P_INVITE_RESULT, // P2P_INVITE_RESULT,
// PermUtils // PermUtils
PERM_REQUEST, PERM_REQUEST,

View file

@ -20,6 +20,7 @@
package org.eehouse.android.xw4; package org.eehouse.android.xw4;
import android.app.Service;
import android.app.Activity; import android.app.Activity;
import android.content.BroadcastReceiver; import android.content.BroadcastReceiver;
import android.content.Context; import android.content.Context;
@ -53,16 +54,33 @@ import java.util.Set;
import org.json.JSONException; import org.json.JSONException;
import org.json.JSONObject; import org.json.JSONObject;
import org.eehouse.android.xw4.jni.CommsAddrRec;
import org.eehouse.android.xw4.jni.CommsAddrRec.CommsConnType;
import org.eehouse.android.xw4.jni.XwJNI;
import junit.framework.Assert; import junit.framework.Assert;
public class WifiDirectService { public class WifiDirectService extends XWService {
private static final String MAC_ADDR_KEY = "p2p_mac_addr"; private static final String MAC_ADDR_KEY = "p2p_mac_addr";
private static final String SERVICE_NAME = "_xwords"; private static final String SERVICE_NAME = "_xwords";
private static final String SERVICE_REG_TYPE = "_presence._tcp"; private static final String SERVICE_REG_TYPE = "_presence._tcp";
private static boolean WIFI_DIRECT_ENABLED = true; private static boolean WIFI_DIRECT_ENABLED = true;
private static final int USE_PORT = 5432; private static final int USE_PORT = 5432;
private static Context sContext; private enum P2PAction { _NONE,
GOT_MSG,
}
private static final String KEY_CMD = "cmd";
private static final String KEY_GAMEID = "gmid";
private static final String KEY_DATA = "data";
private static final String KEY_MAC = "myMac";
private static final String KEY_RETADDR = "raddr";
private static final String CMD_PING = "ping";
private static final String CMD_PONG = "pong";
private static final String CMD_MSG = "msg";
private static Channel sChannel; private static Channel sChannel;
private static ChannelListener sListener; private static ChannelListener sListener;
private static IntentFilter sIntentFilter; private static IntentFilter sIntentFilter;
@ -77,7 +95,39 @@ public class WifiDirectService {
= new HashMap<String, BiDiSockWrap>(); = new HashMap<String, BiDiSockWrap>();
private static Map<String, Long> sPendingDevs = new HashMap<String, Long>(); private static Map<String, Long> sPendingDevs = new HashMap<String, Long>();
private static String sMacAddress; private static String sMacAddress;
public static Activity sActivity;
private P2pMsgSink m_sink;
@Override
public void onCreate()
{
m_sink = new P2pMsgSink();
}
@Override
public int onStartCommand( Intent intent, int flags, int startId )
{
int result;
if ( WIFI_DIRECT_ENABLED ) {
result = Service.START_STICKY;
int ordinal = intent.getIntExtra( KEY_CMD, -1 );
if ( -1 != ordinal ) {
P2PAction cmd = P2PAction.values()[ordinal];
switch ( cmd ) {
case GOT_MSG:
handleGotMessage( intent );
break;
}
}
} else {
result = Service.START_NOT_STICKY;
stopSelf( startId );
}
return result;
}
public static boolean supported() public static boolean supported()
{ {
@ -102,18 +152,30 @@ public class WifiDirectService {
public static int sendPacket( Context context, String macAddr, int gameID, public static int sendPacket( Context context, String macAddr, int gameID,
byte[] buf ) byte[] buf )
{ {
DbgUtils.logd( WifiDirectService.class, "sendPacket(): have %d bytes for %s", int nSent = -1;
buf.length, macAddr );
BiDiSockWrap wrap = sSocketWrapMap.get( macAddr ); BiDiSockWrap wrap = sSocketWrapMap.get( macAddr );
DbgUtils.logd( WifiDirectService.class, "sendPacket(): I %s have a socket for it", if ( null != wrap ) {
(null == wrap ? "DO NOT" : "DO!!" ) ); try {
return -1; JSONObject packet = new JSONObject()
.put( KEY_CMD, CMD_MSG )
.put( KEY_DATA, XwJNI.base64Encode( buf ) )
.put( KEY_GAMEID, gameID )
;
wrap.send( packet );
nSent = buf.length;
} catch ( JSONException jse ) {
DbgUtils.logex( jse );
}
} else {
DbgUtils.logd( WifiDirectService.class, "no socket for packet for %s",
macAddr );
}
return nSent;
} }
public static void activityResumed( Activity activity ) public static void activityResumed( Activity activity )
{ {
if ( WIFI_DIRECT_ENABLED ) { if ( WIFI_DIRECT_ENABLED ) {
sActivity = activity;
if ( initListeners( activity ) ) { if ( initListeners( activity ) ) {
activity.registerReceiver( sReceiver, sIntentFilter ); activity.registerReceiver( sReceiver, sIntentFilter );
DbgUtils.logd( WifiDirectService.class, "activityResumed() done" ); DbgUtils.logd( WifiDirectService.class, "activityResumed() done" );
@ -125,7 +187,6 @@ public class WifiDirectService {
public static void activityPaused( Activity activity ) public static void activityPaused( Activity activity )
{ {
if ( WIFI_DIRECT_ENABLED ) { if ( WIFI_DIRECT_ENABLED ) {
sActivity = null;
Assert.assertNotNull( sReceiver ); Assert.assertNotNull( sReceiver );
// No idea why I'm seeing this exception... // No idea why I'm seeing this exception...
try { try {
@ -142,7 +203,6 @@ public class WifiDirectService {
private static boolean initListeners( final Context context ) private static boolean initListeners( final Context context )
{ {
boolean succeeded = false; boolean succeeded = false;
sContext = context;
if ( WIFI_DIRECT_ENABLED ) { if ( WIFI_DIRECT_ENABLED ) {
if ( null == sListener ) { if ( null == sListener ) {
try { try {
@ -172,8 +232,8 @@ public class WifiDirectService {
if ( nowConnected ) { if ( nowConnected ) {
try { try {
wrap.send( new JSONObject() wrap.send( new JSONObject()
.put( "cmd", "ping" ) .put( KEY_CMD, CMD_PING )
.put( "myMac", getMyMacAddress( context ) ) ); .put( KEY_MAC, getMyMacAddress( context ) ) );
} catch ( JSONException jse ) { } catch ( JSONException jse ) {
DbgUtils.logex( jse ); DbgUtils.logex( jse );
} }
@ -285,15 +345,15 @@ public class WifiDirectService {
DbgUtils.logd( WifiDirectService.class, "tryConnect(%s): already connected", DbgUtils.logd( WifiDirectService.class, "tryConnect(%s): already connected",
macAddress ); macAddress );
} else if ( connectPending( macAddress ) ) { } else if ( connectPending( macAddress ) ) {
// Do nothing
} else { } else {
// sSocketWrapMap.put( macAddress, null );
DbgUtils.logd( WifiDirectService.class, "trying to connect to %s", DbgUtils.logd( WifiDirectService.class, "trying to connect to %s",
device.toString() ); device.toString() );
WifiP2pConfig config = new WifiP2pConfig(); WifiP2pConfig config = new WifiP2pConfig();
config.deviceAddress = device.deviceAddress; config.deviceAddress = device.deviceAddress;
config.wps.setup = WpsInfo.PBC; config.wps.setup = WpsInfo.PBC;
WifiP2pManager mgr = (WifiP2pManager)sContext WifiP2pManager mgr = (WifiP2pManager)XWApp.getContext()
.getSystemService(Context.WIFI_P2P_SERVICE); .getSystemService(Context.WIFI_P2P_SERVICE);
mgr.connect( sChannel, config, new ActionListener() { mgr.connect( sChannel, config, new ActionListener() {
@ -319,17 +379,31 @@ public class WifiDirectService {
private static void storeByAddress( BiDiSockWrap wrap, JSONObject packet ) private static void storeByAddress( BiDiSockWrap wrap, JSONObject packet )
{ {
String macAddress = packet.optString( "myMac", null ); String macAddress = packet.optString( KEY_MAC, null );
// Assert.assertNotNull( macAddress ); // Assert.assertNotNull( macAddress );
if ( null != macAddress ) { if ( null != macAddress ) {
Assert.assertNull( sSocketWrapMap.get( macAddress ) ); Assert.assertNull( sSocketWrapMap.get( macAddress ) );
wrap.setMacAddress( macAddress );
sSocketWrapMap.put( macAddress, wrap ); sSocketWrapMap.put( macAddress, wrap );
DbgUtils.logd( WifiDirectService.class, DbgUtils.logd( WifiDirectService.class,
"processPacket(); storing wrap for %s", "storeByAddress(); storing wrap for %s",
macAddress ); macAddress );
} }
} }
private void handleGotMessage( Intent intent )
{
DbgUtils.logd( getClass(), "processPacket(%s)", intent.toString() );
int gameID = intent.getIntExtra( KEY_GAMEID, 0 );
byte[] data = XwJNI.base64Decode( intent.getStringExtra( KEY_DATA ) );
String macAddress = intent.getStringExtra( KEY_RETADDR );
CommsAddrRec addr = new CommsAddrRec( CommsConnType.COMMS_CONN_P2P )
.setP2PParams( macAddress );
ReceiveResult rslt = receiveMessage( this, gameID, m_sink, data, addr );
}
private static void processPacket( BiDiSockWrap wrap, byte[] bytes ) private static void processPacket( BiDiSockWrap wrap, byte[] bytes )
{ {
String asStr = new String(bytes); String asStr = new String(bytes);
@ -337,23 +411,30 @@ public class WifiDirectService {
try { try {
JSONObject asObj = new JSONObject( asStr ); JSONObject asObj = new JSONObject( asStr );
DbgUtils.logd( WifiDirectService.class, "got json: %s", asObj.toString() ); DbgUtils.logd( WifiDirectService.class, "got json: %s", asObj.toString() );
final String cmd = asObj.optString( "cmd", "" ); final String cmd = asObj.optString( KEY_CMD, "" );
sActivity.runOnUiThread( new Runnable() { if ( cmd.equals( CMD_PING ) ) {
public void run() {
DbgUtils.showf( sActivity, "got cmd: %s", cmd );
}
} );
if ( cmd.equals( "ping" ) ) {
storeByAddress( wrap, asObj ); storeByAddress( wrap, asObj );
try { try {
wrap.send( new JSONObject() wrap.send( new JSONObject()
.put( "cmd", "pong" ) .put( KEY_CMD, CMD_PONG )
.put( "myMac", getMyMacAddress() ) ); .put( KEY_MAC, getMyMacAddress() ) );
} catch ( JSONException jse ) { } catch ( JSONException jse ) {
DbgUtils.logex( jse ); DbgUtils.logex( jse );
} }
} else if ( cmd.equals( "pong" ) ) { } else if ( cmd.equals( CMD_PONG ) ) {
storeByAddress( wrap, asObj ); storeByAddress( wrap, asObj );
} else if ( cmd.equals( CMD_MSG ) ) {
// byte[] data = XwJNI.base64Decode( asObj.optString( KEY_DATA, null ) );
int gameID = asObj.optInt( KEY_GAMEID, 0 );
if ( 0 != gameID ) {
Intent intent = getIntentTo( P2PAction.GOT_MSG );
intent.putExtra( KEY_GAMEID, gameID );
intent.putExtra( KEY_DATA, asObj.optString( KEY_DATA, null ) );
intent.putExtra( KEY_RETADDR, wrap.getMacAddress() );
XWApp.getContext().startService( intent );
} else {
Assert.fail(); // don't ship with this!!!
}
} }
} catch ( JSONException jse ) { } catch ( JSONException jse ) {
DbgUtils.logex( jse ); DbgUtils.logex( jse );
@ -397,6 +478,23 @@ public class WifiDirectService {
} }
} }
private static Intent getIntentTo( P2PAction cmd )
{
Context context = XWApp.getContext();
Intent intent = null;
if ( null != context ) {
intent = new Intent( context, WifiDirectService.class );
intent.putExtra( KEY_CMD, cmd.ordinal() );
} else {
// This basically means we can't receive P2P messages when in the
// background, which is silly. They're coming in on sockets. Do I
// need a socket to have a context associated with it?
DbgUtils.logd( WifiDirectService.class,
"getIntentTo(): contenxt null" );
}
return intent;
}
private static class WFDBroadcastReceiver extends BroadcastReceiver { private static class WFDBroadcastReceiver extends BroadcastReceiver {
private WifiP2pManager mManager; private WifiP2pManager mManager;
private Channel mChannel; private Channel mChannel;
@ -515,4 +613,16 @@ public class WifiDirectService {
// } // }
} }
} }
private class P2pMsgSink extends MultiMsgSink {
public P2pMsgSink() { super( WifiDirectService.this ); }
// @Override
// public int sendViaP2P( byte[] buf, int gameID, CommsAddrRec addr )
// {
// return WifiDirectService
// .sendPacket( m_context, addr.p2p_addr, gameID, buf );
// }
}
} }

View file

@ -48,6 +48,8 @@ public class XWActivity extends FragmentActivity implements Delegator {
super.onCreate( savedInstanceState ); super.onCreate( savedInstanceState );
m_dlgt = dlgt; m_dlgt = dlgt;
Assert.assertTrue( getApplicationContext() == XWApp.getContext() );
int layoutID = m_dlgt.getLayoutID(); int layoutID = m_dlgt.getLayoutID();
if ( 0 < layoutID ) { if ( 0 < layoutID ) {
m_dlgt.setContentView( layoutID ); m_dlgt.setContentView( layoutID );

View file

@ -30,6 +30,8 @@ import org.eehouse.android.xw4.jni.XwJNI;
import java.util.UUID; import java.util.UUID;
import junit.framework.Assert;
public class XWApp extends Application { public class XWApp extends Application {
public static final boolean BTSUPPORTED = true; public static final boolean BTSUPPORTED = true;
@ -60,6 +62,7 @@ public class XWApp extends Application {
public void onCreate() public void onCreate()
{ {
s_context = this; s_context = this;
Assert.assertTrue( s_context == s_context.getApplicationContext() );
super.onCreate(); super.onCreate();
// This one line should always get logged even if logging is // This one line should always get logged even if logging is
@ -122,5 +125,9 @@ public class XWApp extends Application {
return s_onEmulator; return s_onEmulator;
} }
public static Context getContext() { return s_context; } public static Context getContext()
{
Assert.assertNotNull( s_context );
return s_context;
}
} }

View file

@ -270,9 +270,10 @@ public class CommsAddrRec {
sms_port = 1; // so don't assert in comms.... sms_port = 1; // so don't assert in comms....
} }
public void setP2PParams( String macAddress ) public CommsAddrRec setP2PParams( String macAddress )
{ {
p2p_addr = macAddress; p2p_addr = macAddress;
return this;
} }
public void populate( Context context, CommsConnTypeSet newTypes ) public void populate( Context context, CommsConnTypeSet newTypes )