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

2022-09-03 14:24:16

从Android 5.0.0开始,您可以长按WiFi连接并将该连接写入标签(“写入NFC标签”)。您可以在此处找到该操作的源代码:WriteWifiConfigToNfcDialog.java。采用WiFi连接并创建NDEF有效负载的相关行似乎在这里:

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

mWifiManager是 的一个实例,但不是 API 的一部分。通过跟踪此方法,我们可以在此处找到其提交:添加对NFC WSC令牌创建的调用,不幸的是,这无济于事。这就是我的调查已经结束的地方。编辑:我发现了以下调用堆栈:WifiManagergetWpsNfcConfigurationToken

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,密码,加密/身份验证类型的类型。NdefRecordapplication/vnd.wfa.wsc

我当然已经检查了Android创建的实际记录的字节,但是手动重新创建此过程似乎非常不可靠,并且非常乏味。application/vnd.wfa.wsc


答案 1

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

首先,我创建了一个NFC帮助程序类(恰如其分地命名为NFCHelper.java),它具有构造记录所需的所有字节常量。然后,我创建了一个黑客方法,用于创建所需的两个记录之一。该规范实际上在这里相当无用,我所做的是通过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);
} 

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

注意:这是一个相当黑客的实现(您可能会注意到)。我包括AES和AES / TKIP字节,因为我在测试中发现它适用于Android 5下使用不同加密/身份验证方法的各种网络.*请随时更改函数原型,String数组与我正在做的事情配合得很好。

使用在上面的第一个代码段中创建的两条记录,您应该将其传递到 a 中并将其写入您的代码。NdefMessage

不久的将来,我将用图形和东西写一个更好/更强大的soln,所以我会更新这个答案。


答案 2

最终的调用由 wpa_supplicant 模块处理。此处介绍了此功能。我认为可以在wps_supplicant中找到这方面的实际实现。doStringCommand("WPS_NFC_CONFIG_TOKEN WPS " + netId)

你实际上试图做的事情实际上并不是Android特有的。它在“WiFi简单配置技术规范”中定义,您可以通过填写此表格下载。相关部分应为 10.1.2 配置令牌

NfcUtils.java为此提供了一个有效的实现!有一些FIXME和TODO,但总的来说它有效,应该让你对你需要做什么有一个很好的了解。

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


推荐