Support to share Wi-Fi hotspot via QR code
1. QR code scanner (Wi-Fi Easy Connect) does not support sharing Wi-Fi hotspot at current stage 2. Wi-Fi hotspot QR code button only shows when Wi-Fi hotspot is enabled 3. The QR code has the security string "WPA" for hotspot's WPA2_PSK Bug: 123151660 Test: atest WifiTetherSSIDPreferenceControllerTest WifiQrCodeTest WifiDppConfiguratorActivityTest WifiDppEnrolleeActivityTest WifiDppQrCodeGeneratorFragmentTest WifiDppQrCodeScannerFragmentTest WifiNetworkListFragmentTest WifiDppChooseSavedWifiNetworkFragmentTest Change-Id: I2e89450180b82cc841ee3b15be52bfc6f9f6164d
This commit is contained in:
@@ -75,6 +75,7 @@ public class WifiDppConfiguratorActivity extends InstrumentedActivity implements
|
||||
private static final String KEY_WIFI_PRESHARED_KEY = "key_wifi_preshared_key";
|
||||
private static final String KEY_WIFI_HIDDEN_SSID = "key_wifi_hidden_ssid";
|
||||
private static final String KEY_WIFI_NETWORK_ID = "key_wifi_network_id";
|
||||
private static final String KEY_IS_HOTSPOT = "key_is_hotspot";
|
||||
|
||||
private FragmentManager mFragmentManager;
|
||||
|
||||
@@ -104,14 +105,15 @@ public class WifiDppConfiguratorActivity extends InstrumentedActivity implements
|
||||
|
||||
mWifiDppQrCode = WifiQrCode.getValidWifiDppQrCodeOrNull(qrCode);
|
||||
|
||||
String security = savedInstanceState.getString(KEY_WIFI_SECURITY);
|
||||
String ssid = savedInstanceState.getString(KEY_WIFI_SSID);
|
||||
String preSharedKey = savedInstanceState.getString(KEY_WIFI_PRESHARED_KEY);
|
||||
boolean hiddenSsid = savedInstanceState.getBoolean(KEY_WIFI_HIDDEN_SSID);
|
||||
int networkId = savedInstanceState.getInt(KEY_WIFI_NETWORK_ID);
|
||||
final String security = savedInstanceState.getString(KEY_WIFI_SECURITY);
|
||||
final String ssid = savedInstanceState.getString(KEY_WIFI_SSID);
|
||||
final String preSharedKey = savedInstanceState.getString(KEY_WIFI_PRESHARED_KEY);
|
||||
final boolean hiddenSsid = savedInstanceState.getBoolean(KEY_WIFI_HIDDEN_SSID);
|
||||
final int networkId = savedInstanceState.getInt(KEY_WIFI_NETWORK_ID);
|
||||
final boolean isHotspot = savedInstanceState.getBoolean(KEY_IS_HOTSPOT);
|
||||
|
||||
mWifiNetworkConfig = WifiNetworkConfig.getValidConfigOrNull(security, ssid,
|
||||
preSharedKey, hiddenSsid, networkId);
|
||||
preSharedKey, hiddenSsid, networkId, isHotspot);
|
||||
} else {
|
||||
handleIntent(getIntent());
|
||||
}
|
||||
@@ -361,6 +363,7 @@ public class WifiDppConfiguratorActivity extends InstrumentedActivity implements
|
||||
outState.putString(KEY_WIFI_PRESHARED_KEY, mWifiNetworkConfig.getPreSharedKey());
|
||||
outState.putBoolean(KEY_WIFI_HIDDEN_SSID, mWifiNetworkConfig.getHiddenSsid());
|
||||
outState.putInt(KEY_WIFI_NETWORK_ID, mWifiNetworkConfig.getNetworkId());
|
||||
outState.putBoolean(KEY_IS_HOTSPOT, mWifiNetworkConfig.isHotspot());
|
||||
}
|
||||
|
||||
super.onSaveInstanceState(outState);
|
||||
@@ -393,7 +396,8 @@ public class WifiDppConfiguratorActivity extends InstrumentedActivity implements
|
||||
wifiConfiguration.getPrintableSsid(),
|
||||
wifiConfiguration.preSharedKey,
|
||||
/* hiddenSsid */ false,
|
||||
wifiConfiguration.networkId);
|
||||
wifiConfiguration.networkId,
|
||||
/* isHotspot */ false);
|
||||
}
|
||||
}
|
||||
|
||||
|
@@ -86,7 +86,8 @@ public class WifiDppQrCodeGeneratorFragment extends WifiDppQrCodeBaseFragment {
|
||||
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
|
||||
final WifiNetworkConfig wifiNetworkConfig = getWifiNetworkConfigFromHostActivity();
|
||||
MenuItem menuItem;
|
||||
if (wifiNetworkConfig.isSupportWifiDpp(getActivity())) {
|
||||
if (!wifiNetworkConfig.isHotspot() &&
|
||||
wifiNetworkConfig.isSupportWifiDpp(getActivity())) {
|
||||
menuItem = menu.add(0, Menu.FIRST, 0, R.string.next_label);
|
||||
menuItem.setIcon(R.drawable.ic_scan_24dp);
|
||||
menuItem.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS);
|
||||
@@ -127,9 +128,15 @@ public class WifiDppQrCodeGeneratorFragment extends WifiDppQrCodeBaseFragment {
|
||||
setHeaderIconImageResource(R.drawable.ic_qrcode_24dp);
|
||||
|
||||
final WifiNetworkConfig wifiNetworkConfig = getWifiNetworkConfigFromHostActivity();
|
||||
mTitle.setText(R.string.wifi_dpp_share_wifi);
|
||||
mSummary.setText(getString(R.string.wifi_dpp_scan_qr_code_with_another_device,
|
||||
wifiNetworkConfig.getSsid()));
|
||||
if (wifiNetworkConfig.isHotspot()) {
|
||||
mTitle.setText(R.string.wifi_dpp_share_hotspot);
|
||||
mSummary.setText(getString(R.string.wifi_dpp_scan_qr_code_to_share_hotspot,
|
||||
wifiNetworkConfig.getSsid()));
|
||||
} else {
|
||||
mTitle.setText(R.string.wifi_dpp_share_wifi);
|
||||
mSummary.setText(getString(R.string.wifi_dpp_scan_qr_code_with_another_device,
|
||||
wifiNetworkConfig.getSsid()));
|
||||
}
|
||||
|
||||
mQrCode = wifiNetworkConfig.getQrCode();
|
||||
setQrCode();
|
||||
|
@@ -16,11 +16,16 @@
|
||||
|
||||
package com.android.settings.wifi.dpp;
|
||||
|
||||
import android.app.KeyguardManager;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.hardware.biometrics.BiometricPrompt;
|
||||
import android.net.wifi.WifiConfiguration.KeyMgmt;
|
||||
import android.net.wifi.WifiConfiguration;
|
||||
import android.net.wifi.WifiManager;
|
||||
import android.os.CancellationSignal;
|
||||
import android.os.Handler;
|
||||
import android.os.Looper;
|
||||
import android.text.TextUtils;
|
||||
import android.util.FeatureFlagUtils;
|
||||
|
||||
@@ -70,6 +75,9 @@ public class WifiDppUtils {
|
||||
/** The data corresponding to {@code WifiConfiguration} networkId */
|
||||
public static final String EXTRA_WIFI_NETWORK_ID = "networkId";
|
||||
|
||||
/** The data to recognize if it's a Wi-Fi hotspot for configuration */
|
||||
public static final String EXTRA_IS_HOTSPOT = "isHotspot";
|
||||
|
||||
/** Used by {@link android.provider.Settings#ACTION_PROCESS_WIFI_EASY_CONNECT_URI} to
|
||||
* indicate test mode UI should be shown. Test UI does not make API calls. Value is a boolean.*/
|
||||
public static final String EXTRA_TEST = "test";
|
||||
@@ -142,24 +150,12 @@ public class WifiDppUtils {
|
||||
return str.substring(begin, end+1);
|
||||
}
|
||||
|
||||
private static String getSecurityString(AccessPoint accessPoint) {
|
||||
switch(accessPoint.getSecurity()) {
|
||||
case AccessPoint.SECURITY_WEP:
|
||||
return WifiQrCode.SECURITY_WEP;
|
||||
case AccessPoint.SECURITY_PSK:
|
||||
return WifiQrCode.SECURITY_WPA_PSK;
|
||||
case AccessPoint.SECURITY_SAE:
|
||||
return WifiQrCode.SECURITY_SAE;
|
||||
default:
|
||||
return WifiQrCode.SECURITY_NO_PASSWORD;
|
||||
}
|
||||
}
|
||||
|
||||
static String getSecurityString(WifiConfiguration config) {
|
||||
if (config.allowedKeyManagement.get(KeyMgmt.SAE)) {
|
||||
return WifiQrCode.SECURITY_SAE;
|
||||
}
|
||||
if (config.allowedKeyManagement.get(KeyMgmt.WPA_PSK)) {
|
||||
if (config.allowedKeyManagement.get(KeyMgmt.WPA_PSK) ||
|
||||
config.allowedKeyManagement.get(KeyMgmt.WPA2_PSK)) {
|
||||
return WifiQrCode.SECURITY_WPA_PSK;
|
||||
}
|
||||
return (config.wepKeys[0] == null) ?
|
||||
@@ -171,6 +167,9 @@ public class WifiDppUtils {
|
||||
* security. It may return null if the security is not supported by QR code generator nor
|
||||
* scanner.
|
||||
*
|
||||
* Do not use this method for Wi-Fi hotspot network, use
|
||||
* {@code getHotspotConfiguratorIntentOrNull} instead.
|
||||
*
|
||||
* @param context The context to use for the content resolver
|
||||
* @param wifiManager An instance of {@link WifiManager}
|
||||
* @param accessPoint An instance of {@link AccessPoint}
|
||||
@@ -187,15 +186,63 @@ public class WifiDppUtils {
|
||||
return null;
|
||||
}
|
||||
|
||||
final WifiConfiguration wifiConfig = accessPoint.getConfig();
|
||||
final String ssid = removeFirstAndLastDoubleQuotes(wifiConfig.SSID);
|
||||
final String security = getSecurityString(accessPoint);
|
||||
String preSharedKey = wifiConfig.preSharedKey;
|
||||
final WifiConfiguration wifiConfiguration = accessPoint.getConfig();
|
||||
setConfiguratorIntentExtra(intent, wifiManager, wifiConfiguration);
|
||||
|
||||
if (wifiConfiguration.networkId == WifiConfiguration.INVALID_NETWORK_ID) {
|
||||
throw new IllegalArgumentException("Invalid network ID");
|
||||
} else {
|
||||
intent.putExtra(EXTRA_WIFI_NETWORK_ID, wifiConfiguration.networkId);
|
||||
}
|
||||
|
||||
return intent;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an intent to launch QR code generator for the Wi-Fi hotspot. It may return null if
|
||||
* the security is not supported by QR code generator.
|
||||
*
|
||||
* @param context The context to use for the content resolver
|
||||
* @param wifiManager An instance of {@link WifiManager}
|
||||
* @param wifiConfiguration {@link WifiConfiguration} of the Wi-Fi hotspot
|
||||
* @return Intent for launching QR code generator
|
||||
*/
|
||||
public static Intent getHotspotConfiguratorIntentOrNull(Context context,
|
||||
WifiManager wifiManager, WifiConfiguration wifiConfiguration) {
|
||||
final Intent intent = new Intent(context, WifiDppConfiguratorActivity.class);
|
||||
if (isSupportHotspotConfiguratorQrCodeGenerator(wifiConfiguration)) {
|
||||
intent.setAction(WifiDppConfiguratorActivity.ACTION_CONFIGURATOR_QR_CODE_GENERATOR);
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
|
||||
setConfiguratorIntentExtra(intent, wifiManager, wifiConfiguration);
|
||||
|
||||
intent.putExtra(EXTRA_WIFI_NETWORK_ID, WifiConfiguration.INVALID_NETWORK_ID);
|
||||
intent.putExtra(EXTRA_IS_HOTSPOT, true);
|
||||
|
||||
return intent;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set all extra except {@code EXTRA_WIFI_NETWORK_ID} for the intent to
|
||||
* launch configurator activity later.
|
||||
*
|
||||
* @param intent the target to set extra
|
||||
* @param wifiManager an instance of {@code WifiManager}
|
||||
* @param wifiConfiguration the Wi-Fi network for launching configurator activity
|
||||
*/
|
||||
private static void setConfiguratorIntentExtra(Intent intent, WifiManager wifiManager,
|
||||
WifiConfiguration wifiConfiguration) {
|
||||
final String ssid = removeFirstAndLastDoubleQuotes(wifiConfiguration.SSID);
|
||||
final String security = getSecurityString(wifiConfiguration);
|
||||
String preSharedKey = wifiConfiguration.preSharedKey;
|
||||
|
||||
if (preSharedKey != null) {
|
||||
// When the value of this key is read, the actual key is not returned, just a "*".
|
||||
// Call privileged system API to obtain actual key.
|
||||
preSharedKey = removeFirstAndLastDoubleQuotes(getPresharedKey(wifiManager, wifiConfig));
|
||||
preSharedKey = removeFirstAndLastDoubleQuotes(getPresharedKey(wifiManager,
|
||||
wifiConfiguration));
|
||||
}
|
||||
|
||||
if (!TextUtils.isEmpty(ssid)) {
|
||||
@@ -207,13 +254,6 @@ public class WifiDppUtils {
|
||||
if (!TextUtils.isEmpty(preSharedKey)) {
|
||||
intent.putExtra(EXTRA_WIFI_PRE_SHARED_KEY, preSharedKey);
|
||||
}
|
||||
if (wifiConfig.networkId == WifiConfiguration.INVALID_NETWORK_ID) {
|
||||
throw new IllegalArgumentException("Invalid network ID");
|
||||
} else {
|
||||
intent.putExtra(EXTRA_WIFI_NETWORK_ID, wifiConfig.networkId);
|
||||
}
|
||||
|
||||
return intent;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -228,6 +268,54 @@ public class WifiDppUtils {
|
||||
isSupportConfiguratorQrCodeGenerator(accessPoint);
|
||||
}
|
||||
|
||||
/**
|
||||
* Shows authentication screen to confirm credentials (pin, pattern or password) for the current
|
||||
* user of the device.
|
||||
*
|
||||
* @param context The {@code Context} used to get {@code KeyguardManager} service
|
||||
* @param title The title on lock screen
|
||||
* @param description The description on lock screen
|
||||
* @param successRunnable The {@code Runnable} which will be executed if the user does not setup
|
||||
* device security or if lock screen is unlocked
|
||||
*/
|
||||
public static void showLockScreen(Context context, String title, String description,
|
||||
Runnable successRunnable) {
|
||||
final KeyguardManager keyguardManager = (KeyguardManager) context.getSystemService(
|
||||
Context.KEYGUARD_SERVICE);
|
||||
|
||||
if (keyguardManager.isKeyguardSecure()) {
|
||||
final BiometricPrompt.AuthenticationCallback authenticationCallback =
|
||||
new BiometricPrompt.AuthenticationCallback() {
|
||||
@Override
|
||||
public void onAuthenticationSucceeded(
|
||||
BiometricPrompt.AuthenticationResult result) {
|
||||
successRunnable.run();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onAuthenticationError(int errorCode, CharSequence errString) {
|
||||
//Do nothing
|
||||
}
|
||||
};
|
||||
|
||||
final BiometricPrompt.Builder builder = new BiometricPrompt.Builder(context)
|
||||
.setTitle(title)
|
||||
.setDescription(description);
|
||||
|
||||
if (keyguardManager.isDeviceSecure()) {
|
||||
builder.setDeviceCredentialAllowed(true);
|
||||
}
|
||||
|
||||
final BiometricPrompt bp = builder.build();
|
||||
final Handler handler = new Handler(Looper.getMainLooper());
|
||||
bp.authenticate(new CancellationSignal(),
|
||||
runnable -> handler.post(runnable),
|
||||
authenticationCallback);
|
||||
} else {
|
||||
successRunnable.run();
|
||||
}
|
||||
}
|
||||
|
||||
private static boolean isSupportConfiguratorQrCodeScanner(Context context,
|
||||
AccessPoint accessPoint) {
|
||||
if (!isWifiDppEnabled(context)) {
|
||||
@@ -254,4 +342,13 @@ public class WifiDppUtils {
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
private static boolean isSupportHotspotConfiguratorQrCodeGenerator(
|
||||
WifiConfiguration wifiConfiguration) {
|
||||
// QR code generator produces QR code with ZXing's Wi-Fi network config format,
|
||||
// it supports PSK and WEP and non security
|
||||
// KeyMgmt.NONE is for WEP or non security
|
||||
return wifiConfiguration.allowedKeyManagement.get(KeyMgmt.WPA2_PSK) ||
|
||||
wifiConfiguration.allowedKeyManagement.get(KeyMgmt.NONE);
|
||||
}
|
||||
}
|
||||
|
@@ -30,7 +30,6 @@ import android.net.wifi.WifiManager;
|
||||
import android.text.TextUtils;
|
||||
import android.util.Log;
|
||||
|
||||
import androidx.annotation.Keep;
|
||||
import androidx.annotation.VisibleForTesting;
|
||||
|
||||
/**
|
||||
@@ -52,15 +51,17 @@ public class WifiNetworkConfig {
|
||||
private String mPreSharedKey;
|
||||
private boolean mHiddenSsid;
|
||||
private int mNetworkId;
|
||||
private boolean mIsHotspot;
|
||||
|
||||
@VisibleForTesting
|
||||
WifiNetworkConfig(String security, String ssid, String preSharedKey,
|
||||
boolean hiddenSsid, int networkId) {
|
||||
boolean hiddenSsid, int networkId, boolean isHotspot) {
|
||||
mSecurity = security;
|
||||
mSsid = ssid;
|
||||
mPreSharedKey = preSharedKey;
|
||||
mHiddenSsid = hiddenSsid;
|
||||
mNetworkId = networkId;
|
||||
mIsHotspot = isHotspot;
|
||||
}
|
||||
|
||||
public WifiNetworkConfig(WifiNetworkConfig config) {
|
||||
@@ -69,6 +70,7 @@ public class WifiNetworkConfig {
|
||||
mPreSharedKey = config.mPreSharedKey;
|
||||
mHiddenSsid = config.mHiddenSsid;
|
||||
mNetworkId = config.mNetworkId;
|
||||
mIsHotspot = config.mIsHotspot;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -86,23 +88,26 @@ public class WifiNetworkConfig {
|
||||
* android.settings.WIFI_DPP_CONFIGURATOR_QR_CODE_SCANNER
|
||||
*/
|
||||
public static WifiNetworkConfig getValidConfigOrNull(Intent intent) {
|
||||
String security = intent.getStringExtra(WifiDppUtils.EXTRA_WIFI_SECURITY);
|
||||
String ssid = intent.getStringExtra(WifiDppUtils.EXTRA_WIFI_SSID);
|
||||
String preSharedKey = intent.getStringExtra(WifiDppUtils.EXTRA_WIFI_PRE_SHARED_KEY);
|
||||
boolean hiddenSsid = intent.getBooleanExtra(WifiDppUtils.EXTRA_WIFI_HIDDEN_SSID, false);
|
||||
int networkId = intent.getIntExtra(WifiDppUtils.EXTRA_WIFI_NETWORK_ID,
|
||||
final String security = intent.getStringExtra(WifiDppUtils.EXTRA_WIFI_SECURITY);
|
||||
final String ssid = intent.getStringExtra(WifiDppUtils.EXTRA_WIFI_SSID);
|
||||
final String preSharedKey = intent.getStringExtra(WifiDppUtils.EXTRA_WIFI_PRE_SHARED_KEY);
|
||||
final boolean hiddenSsid = intent.getBooleanExtra(WifiDppUtils.EXTRA_WIFI_HIDDEN_SSID,
|
||||
false);
|
||||
final int networkId = intent.getIntExtra(WifiDppUtils.EXTRA_WIFI_NETWORK_ID,
|
||||
WifiConfiguration.INVALID_NETWORK_ID);
|
||||
final boolean isHotspot = intent.getBooleanExtra(WifiDppUtils.EXTRA_IS_HOTSPOT, false);
|
||||
|
||||
return getValidConfigOrNull(security, ssid, preSharedKey, hiddenSsid, networkId);
|
||||
return getValidConfigOrNull(security, ssid, preSharedKey, hiddenSsid, networkId, isHotspot);
|
||||
}
|
||||
|
||||
public static WifiNetworkConfig getValidConfigOrNull(String security, String ssid,
|
||||
String preSharedKey, boolean hiddenSsid, int networkId) {
|
||||
String preSharedKey, boolean hiddenSsid, int networkId, boolean isHotspot) {
|
||||
if (!isValidConfig(security, ssid, preSharedKey, hiddenSsid)) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return new WifiNetworkConfig(security, ssid, preSharedKey, hiddenSsid, networkId);
|
||||
return new WifiNetworkConfig(security, ssid, preSharedKey, hiddenSsid, networkId,
|
||||
isHotspot);
|
||||
}
|
||||
|
||||
public static boolean isValidConfig(WifiNetworkConfig config) {
|
||||
@@ -174,31 +179,30 @@ public class WifiNetworkConfig {
|
||||
return barcode;
|
||||
}
|
||||
|
||||
@Keep
|
||||
public String getSecurity() {
|
||||
return mSecurity;
|
||||
}
|
||||
|
||||
@Keep
|
||||
public String getSsid() {
|
||||
return mSsid;
|
||||
}
|
||||
|
||||
@Keep
|
||||
public String getPreSharedKey() {
|
||||
return mPreSharedKey;
|
||||
}
|
||||
|
||||
@Keep
|
||||
public boolean getHiddenSsid() {
|
||||
return mHiddenSsid;
|
||||
}
|
||||
|
||||
@Keep
|
||||
public int getNetworkId() {
|
||||
return mNetworkId;
|
||||
}
|
||||
|
||||
public boolean isHotspot() {
|
||||
return mIsHotspot;
|
||||
}
|
||||
|
||||
public void connect(Context context, WifiManager.ActionListener listener) {
|
||||
WifiConfiguration wifiConfiguration = getWifiConfigurationOrNull();
|
||||
if (wifiConfiguration == null) {
|
||||
|
@@ -218,7 +218,7 @@ public class WifiNetworkListFragment extends SettingsPreferenceFragment implemen
|
||||
final WifiNetworkConfig networkConfig = WifiNetworkConfig.getValidConfigOrNull(
|
||||
selectedAccessPoint.getSecurityString(/* concise */ true),
|
||||
wifiConfig.getPrintableSsid(), wifiConfig.preSharedKey, /* hiddenSsid */ false,
|
||||
wifiConfig.networkId);
|
||||
wifiConfig.networkId, /* isHotspot */ false);
|
||||
if (mOnChooseNetworkListener != null) {
|
||||
mOnChooseNetworkListener.onChooseNetwork(networkConfig);
|
||||
}
|
||||
@@ -232,7 +232,8 @@ public class WifiNetworkListFragment extends SettingsPreferenceFragment implemen
|
||||
/* ssid */ WifiNetworkConfig.FAKE_SSID,
|
||||
/* preSharedKey */ WifiNetworkConfig.FAKE_PASSWORD,
|
||||
/* hiddenSsid */ true,
|
||||
/* networkId */ WifiConfiguration.INVALID_NETWORK_ID));
|
||||
/* networkId */ WifiConfiguration.INVALID_NETWORK_ID,
|
||||
/* isHotspot*/ false));
|
||||
}
|
||||
} else {
|
||||
return super.onPreferenceTreeClick(preference);
|
||||
|
@@ -133,7 +133,7 @@ public class WifiQrCode {
|
||||
password = removeBackSlash(password);
|
||||
|
||||
mWifiNetworkConfig = WifiNetworkConfig.getValidConfigOrNull(security, ssid, password,
|
||||
hiddenSsid, WifiConfiguration.INVALID_NETWORK_ID);
|
||||
hiddenSsid, WifiConfiguration.INVALID_NETWORK_ID, /* isHotspot */ false);
|
||||
|
||||
if (mWifiNetworkConfig == null) {
|
||||
throw new IllegalArgumentException("Invalid format");
|
||||
|
Reference in New Issue
Block a user