从Android 5.0.0开始,您可以长时间点击WiFi连接并将该连接写入标签("写入NFC标签")。您可以在此处找到该操作的来源:WriteWifiConfigToNfcDialog.java。采用WiFi连接并创建NDEF有效负载的相关线路似乎在此处:
String wpsNfcConfigurationToken = mWifiManager.getWpsNfcConfigurationToken(mAccessPoint.networkId);
mWifiManager
是WifiManager
的实例,但getWpsNfcConfigurationToken
不是API的一部分。通过跟踪此方法,我们可以在此处找到它的提交:Add calls for NFC WSC token creation,遗憾的是没有帮助。 这是我的调查已经用完的地方。编辑:
我发现了以下调用堆栈:
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
的方法,该方法属于application/vnd.wfa.wsc
,并提供SSID,密码,加密/验证类型。
我当然检查了由Android创建的实际var selectedUsers = [];
$(tab).find(".cb:checked").each(function() {
// If the QID is not already in the array, add it
if(jQuery.inArray($(this).val(), selectedUsers) === -1) {
selectedUsers.push({
"qid": $(this).val(),
"first": $(this).data('emp_first'),
"last": $(this).data('emp_last'),
"department": $(this).data('emp_dept')
});
}
});
记录的字节,但手动重新创建这个字节的过程似乎可能非常不可靠,而且非常繁琐。
答案 0 :(得分:10)
答案在于Wi-Fi联盟" Wi-Fi简单配置技术规范v2.0.5" (可供下载here)。 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.
创建配置令牌的方法:
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);
}
许可证和要点here。你可以在网上的任何地方找到concat
方法的实现,或者只编写自己的方法。
注意:这是一个非常糟糕的实现(您可能会注意到)。我在测试中发现了AES和AES / TKIP字节,它在Android 5下使用不同的加密/验证方法为各种网络工作。* 请随意更改函数原型,String数组与我正在做的很好地配合。
使用上面第一个代码段中创建的两个记录,然后您应该将其传递到NdefMessage
并将其写入您的代码。
有一天我很快就会做一个写作和一个更好/更强大的图形和东西的解决方案,所以我会更新这个答案。
答案 1 :(得分:7)
最后doStringCommand("WPS_NFC_CONFIG_TOKEN WPS " + netId)
的调用由 wpa_supplicant 模块处理。此功能描述为here。我认为实际的实现可以在wps_supplicant.c。
你实际上要做的事情并不是Android特有的。它在&#34; WiFi简单配置技术规范&#34;中定义,您可以通过填写this form下载。相关部分应为 10.1.2配置令牌。
NfcUtils.java有一个有效的实施方案!有一些FIXME和TODO,但总的来说它起作用,应该让你很好地了解你需要做什么。
如果您想自己解析此类NdefRecords并使用SSID和密钥执行某些操作,NfcWifiProtectedSetup.java会说明如何执行此操作。