在Android中使用application/vnd.wfa.wsc创建一个NDEF WiFi记录
Creating an NDEF WiFi record using application/vnd.wfa.wsc in Android
从 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 进行系统调用(在任何地方都找不到此代码)。
现在调查结束。
希望有人可以介入并向我展示一种方法,该方法可以在给定 SSID、密码、加密/身份验证类型的情况下创建 application/vnd.wfa.wsc 类型的 NdefRecord。
我当然检查了由 Android 创建的实际 application/vnd.wfa.wsc 记录的字节,但是使用字节手动重新创建这个过程似乎非常不可靠并且非常乏味。
答案在 Wi-Fi 联盟《Wi-Fi 简单配置技术规范 v2.0.5》(可在此处下载)。 Android 使用这种标准格式来配置 WiFi 网络,我错误地认为它是专有的。
首先,我创建了一个 NFC 帮助器类(恰当地命名为 NFCHelper.java),它包含构造记录所需的所有字节常量。然后,我创建了一个 hacky 方法来创建所需的两条记录之一。该规范实际上在这里毫无用处,我所做的是检查了许多已通过 Android 操作系统成功配置的标签的有效负载。最后,您需要有一种机制来预先添加"切换选择记录(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);
}
创建配置令牌的方法:
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);
}
许可证和要点在这里。您可以在网上的任何地方找到 concat 方法的实现,或者自己编写。
注意:这是一个相当老套的实现(您可能会注意到)。我包括了 AES 和 AES/TKIP 字节,因为我在测试中发现它适用于在 Android 5 下使用不同加密/身份验证方法的各种网络。*
请随意更改函数原型,字符串数组与我正在做的工作很好。
使用上面第一个片段中创建的两条记录,您应该将其传递到 NdefMessage 并将其写入您的标签。
很快有一天,我会写一篇文章,并用图形和其他东西写一个更好/更强大的解决方案,所以我会更新这个答案。
doStringCommand("WPS_NFC_CONFIG_TOKEN WPS" + netId) 的调用最终由 wpa_supplicant 模块处理。此处描述了此功能。我认为这个的实际实现可以在 wps_supplicant.c.
中找到
您实际尝试做的并不是 Android 特定的事情。它在"WiFi 简单配置技术规范"中有定义,您可以通过填写此表下载。相关部分应该是 10.1.2 Configuration Token.
NfcUtils.java 对此有一个有效的实现!有一些 FIXME 和 TODO,但总的来说它是有效的,并且应该让您很好地了解您需要做什么。
如果你想自己解析这样的 NdefRecords 并使用 SSID 和密钥做一些事情,NfcWifiProtectedSetup.java 会展示如何做到这一点。