使用Android中的application/vnd.wfa.wsc创建NDEF WiFi记录

hen*_*cjc 15 java android nfc wifi ndef

从Android 5.0.0开始,您可以长时间点击WiFi连接并将该连接写入标签("写入NFC标签").您可以在此处找到该操作的源:WriteWifiConfigToNfcDialog.java.采用WiFi连接并创建NDEF有效负载的相关线路似乎在这里:

String wpsNfcConfigurationToken = mWifiManager.getWpsNfcConfigurationToken(mAccessPoint.networkId);

mWifiManager是一个实例WifiManager,但getWpsNfcConfigurationToken不是API的一部分.通过跟踪此方法,我们可以在此处找到它的提交:添加对NFC WSC令牌创建的调用,遗憾的是没有帮助.这是我的调查已经用完的地方.编辑:我发现了以下调用堆栈:

WifiServiceImpl.java 电话 mWifiStateMachine.syncGetWpsNfcConfigurationToken(netId);

WifiStateMachine.java 电话 mWifiNative.getNfcWpsConfigurationToken(netId);

WifiNative.java 终于有了方法

public String getNfcWpsConfigurationToken(int netId) { return doStringCommand("WPS_NFC_CONFIG_TOKEN WPS " + netId); }

然后调用

String result = doStringCommandNative(mInterfacePrefix + command);

在哪里doStringCommandNative进行系统调用(无法在任何地方找到此代码).

现在调查结束的地方.

希望有人能介入,并告诉我,创建一个方法NdefRecord是类型的application/vnd.wfa.wsc给定的SSID,密码,加密/身份验证类型.

我当然检查了application/vnd.wfa.wsc由Android创建的实际记录的字节,但手动重新创建这个字节的过程似乎可能非常不可靠,而且非常繁琐.

hen*_*cjc 10

答案在于Wi-Fi联盟"Wi-Fi简单配置技术规范v2.0.5"(可从此处下载).Android使用这种标准格式来配置WiFi网络,我错误地认为它是专有的.

首先,我创建了一个NFC辅助类(恰当地命名为NFCHelper.java),它具有构造记录所需的所有字节常量.然后,我创建了一个hacky方法来创建所需的两个记录之一.这里的规范实际上相当无用,我所做的是检查了通过Android OS成功配置的许多标签有效负载.最后,您需要有一种机制来预先设置"切换选择记录(NFC WKT Hs)"(请参阅​​WiFi规范第90页).我相信这条记录"告诉"Android在以下令牌中注册网络.

如何创建移交记录:

ndefRecords = new NdefRecord[2];
byte[] version = new byte[] { (0x1 << 4) | (0x2)};
ndefRecords[0] = new NdefRecord(NdefRecord.TNF_WELL_KNOWN, NdefRecord.RTD_HANDOVER_REQUEST, new byte[0], version);
// and then obviously add the record you create with the method below.
Run Code Online (Sandbox Code Playgroud)

创建配置令牌的方法:

private NdefRecord createWifiRecord(String[] data) {
    String ssid = data[0];
    String password = data[1];
    String auth = data[2];
    String crypt = data[3];
    byte[] authByte = getAuthBytes(auth);
    byte[] cryptByte = getCryptBytes(crypt);
    byte[] ssidByte = ssid.getBytes();
    byte[] passwordByte = password.getBytes();
    byte[] ssidLength = {(byte)((int)Math.floor(ssid.length()/256)), (byte)(ssid.length()%256)};
    byte[] passwordLength = {(byte)((int)Math.floor(password.length()/256)), (byte)(password.length()%256)};
    byte[] cred = {0x00, 0x36};
    byte[] idx = {0x00, 0x01, 0x01};
    byte[] mac = {0x00, 0x06};
    byte[] keypad = {0x00, 0x0B};

    byte[] payload = concat(NFCHelper.CREDENTIAL, cred,
            NFCHelper.NETWORK_IDX, idx,
            NFCHelper.NETWORK_NAME, ssidLength, ssidByte,
            NFCHelper.AUTH_TYPE, NFCHelper.AUTH_WPA_PERSONAL, authByte,
            NFCHelper.CRYPT_TYPE, NFCHelper.CRYPT_WEP, NFCHelper.CRYPT_AES_TKIP,
            NFCHelper.NETWORK_KEY, passwordLength, passwordByte);
           // NFCHelper.MAC_ADDRESS, mac);
    return NdefRecord.createMime(NFC_TOKEN_MIME_TYPE, payload);
} 
Run Code Online (Sandbox Code Playgroud)

许可证和要点在这里.你可以concat在网上任何地方找到该方法的实现,或者只编写自己的方法.

注意:这是一个相当hacky实现(你可能会注意到).我在测试中发现了AES和AES/TKIP字节,它在Android 5下使用不同的加密/验证方法在各种网络中工作.*请随意更改函数原型,String数组刚好用什么工作我在做.

使用上面第一个片段中创建的两个记录,然后您应该将其传递给a NdefMessage并将其写入您的标记.

有一天我很快就会做一个写作和一个更好/更强大的图形和东西的解决方案,所以我会更新这个答案.


noo*_*one 7

最后的调用doStringCommand("WPS_NFC_CONFIG_TOKEN WPS " + netId)wpa_supplicant模块处理.此功能在此处描述.我认为可以在wps_supplicant.c中找到实际的实现.

你实际上要做的不是Android特定的东西.它在"WiFi简单配置技术规范"中定义,您可以通过填写此表单进行下载.相关部分应为10.1.2配置令牌.

NfcUtils.java有一个可行的实现!有一些FIXME和TODO,但总的来说它起作用,应该让你很好地了解你需要做什么.

如果您想自己解析此类NdefRecords并使用SSID和密钥执行某些操作,NfcWifiProtectedSetup.java会显示如何执行此操作.