1 /* 2 * Copyright (C) 2021 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.net; 18 19 import android.annotation.IntDef; 20 import android.annotation.NonNull; 21 import android.annotation.SystemApi; 22 import android.os.Bundle; 23 import android.os.Parcelable; 24 25 import java.lang.annotation.Retention; 26 import java.lang.annotation.RetentionPolicy; 27 import java.util.Collections; 28 import java.util.HashMap; 29 import java.util.Map; 30 import java.util.Objects; 31 32 /** 33 * Network preferences to set the default active network on a per-application basis as per a given 34 * {@link OemNetworkPreference}. An example of this would be to set an application's network 35 * preference to {@link #OEM_NETWORK_PREFERENCE_OEM_PAID_NO_FALLBACK} which would have the default 36 * network for that application set to an unmetered network first if available and if not, it then 37 * set that application's default network to an OEM managed network if available. 38 * 39 * @hide 40 */ 41 @SystemApi 42 public final class OemNetworkPreferences implements Parcelable { 43 // Valid production preferences must be > 0, negative values reserved for testing 44 /** 45 * This preference is only to be used for testing and nothing else. 46 * Use only TRANSPORT_TEST transport networks. 47 * @hide 48 */ 49 public static final int OEM_NETWORK_PREFERENCE_TEST_ONLY = -2; 50 51 /** 52 * This preference is only to be used for testing and nothing else. 53 * If an unmetered network is available, use it. 54 * Otherwise, if a network with the TRANSPORT_TEST transport is available, use it. 55 * Otherwise, use the general default network. 56 * @hide 57 */ 58 public static final int OEM_NETWORK_PREFERENCE_TEST = -1; 59 60 /** 61 * Default in case this value is not set. Using it will result in an error. 62 */ 63 public static final int OEM_NETWORK_PREFERENCE_UNINITIALIZED = 0; 64 65 /** 66 * If an unmetered network is available, use it. 67 * Otherwise, if a network with the OEM_PAID capability is available, use it. 68 * Otherwise, use the general default network. 69 */ 70 public static final int OEM_NETWORK_PREFERENCE_OEM_PAID = 1; 71 72 /** 73 * If an unmetered network is available, use it. 74 * Otherwise, if a network with the OEM_PAID capability is available, use it. 75 * Otherwise, the app doesn't get a default network. 76 */ 77 public static final int OEM_NETWORK_PREFERENCE_OEM_PAID_NO_FALLBACK = 2; 78 79 /** 80 * Use only NET_CAPABILITY_OEM_PAID networks. 81 */ 82 public static final int OEM_NETWORK_PREFERENCE_OEM_PAID_ONLY = 3; 83 84 /** 85 * Use only NET_CAPABILITY_OEM_PRIVATE networks. 86 */ 87 public static final int OEM_NETWORK_PREFERENCE_OEM_PRIVATE_ONLY = 4; 88 89 /** 90 * The max allowed value for an OEM network preference. 91 * @hide 92 */ 93 public static final int OEM_NETWORK_PREFERENCE_MAX = OEM_NETWORK_PREFERENCE_OEM_PRIVATE_ONLY; 94 95 @NonNull 96 private final Bundle mNetworkMappings; 97 98 /** 99 * Return whether this object is empty. 100 * @hide 101 */ 102 public boolean isEmpty() { 103 return mNetworkMappings.keySet().size() == 0; 104 } 105 106 /** 107 * Return the currently built application package name to {@link OemNetworkPreference} mappings. 108 * @return the current network preferences map. 109 */ 110 @NonNull 111 public Map<String, Integer> getNetworkPreferences() { 112 return convertToUnmodifiableMap(mNetworkMappings); 113 } 114 115 private OemNetworkPreferences(@NonNull final Bundle networkMappings) { 116 Objects.requireNonNull(networkMappings); 117 mNetworkMappings = (Bundle) networkMappings.clone(); 118 } 119 120 @Override 121 public String toString() { 122 return "OemNetworkPreferences{" + "mNetworkMappings=" + getNetworkPreferences() + '}'; 123 } 124 125 @Override 126 public boolean equals(Object o) { 127 if (this == o) return true; 128 if (o == null || getClass() != o.getClass()) return false; 129 OemNetworkPreferences that = (OemNetworkPreferences) o; 130 131 return mNetworkMappings.size() == that.mNetworkMappings.size() 132 && mNetworkMappings.toString().equals(that.mNetworkMappings.toString()); 133 } 134 135 @Override 136 public int hashCode() { 137 return Objects.hash(mNetworkMappings); 138 } 139 140 /** 141 * Builder used to create {@link OemNetworkPreferences} objects. Specify the preferred Network 142 * to package name mappings. 143 */ 144 public static final class Builder { 145 private final Bundle mNetworkMappings; 146 147 public Builder() { 148 mNetworkMappings = new Bundle(); 149 } 150 151 /** 152 * Constructor to populate the builder's values with an already built 153 * {@link OemNetworkPreferences}. 154 * @param preferences the {@link OemNetworkPreferences} to populate with. 155 */ 156 public Builder(@NonNull final OemNetworkPreferences preferences) { 157 Objects.requireNonNull(preferences); 158 mNetworkMappings = (Bundle) preferences.mNetworkMappings.clone(); 159 } 160 161 /** 162 * Add a network preference for a given package. Previously stored values for the given 163 * package will be overwritten. 164 * 165 * @param packageName full package name (e.g.: "com.google.apps.contacts") of the app 166 * to use the given preference 167 * @param preference the desired network preference to use 168 * @return The builder to facilitate chaining. 169 */ 170 @NonNull 171 public Builder addNetworkPreference(@NonNull final String packageName, 172 @OemNetworkPreference final int preference) { 173 Objects.requireNonNull(packageName); 174 mNetworkMappings.putInt(packageName, preference); 175 return this; 176 } 177 178 /** 179 * Remove a network preference for a given package. 180 * 181 * @param packageName full package name (e.g.: "com.google.apps.contacts") of the app to 182 * remove a preference for. 183 * @return The builder to facilitate chaining. 184 */ 185 @NonNull 186 public Builder clearNetworkPreference(@NonNull final String packageName) { 187 Objects.requireNonNull(packageName); 188 mNetworkMappings.remove(packageName); 189 return this; 190 } 191 192 /** 193 * Build {@link OemNetworkPreferences} return the current OEM network preferences. 194 */ 195 @NonNull 196 public OemNetworkPreferences build() { 197 return new OemNetworkPreferences(mNetworkMappings); 198 } 199 } 200 201 private static Map<String, Integer> convertToUnmodifiableMap(@NonNull final Bundle bundle) { 202 final Map<String, Integer> networkPreferences = new HashMap<>(); 203 for (final String key : bundle.keySet()) { 204 networkPreferences.put(key, bundle.getInt(key)); 205 } 206 return Collections.unmodifiableMap(networkPreferences); 207 } 208 209 /** @hide */ 210 @IntDef(prefix = "OEM_NETWORK_PREFERENCE_", value = { 211 OEM_NETWORK_PREFERENCE_TEST_ONLY, 212 OEM_NETWORK_PREFERENCE_TEST, 213 OEM_NETWORK_PREFERENCE_UNINITIALIZED, 214 OEM_NETWORK_PREFERENCE_OEM_PAID, 215 OEM_NETWORK_PREFERENCE_OEM_PAID_NO_FALLBACK, 216 OEM_NETWORK_PREFERENCE_OEM_PAID_ONLY, 217 OEM_NETWORK_PREFERENCE_OEM_PRIVATE_ONLY 218 }) 219 @Retention(RetentionPolicy.SOURCE) 220 public @interface OemNetworkPreference {} 221 222 /** 223 * Return the string value for OemNetworkPreference 224 * 225 * @param value int value of OemNetworkPreference 226 * @return string version of OemNetworkPreference 227 * 228 * @hide 229 */ 230 @NonNull 231 public static String oemNetworkPreferenceToString(@OemNetworkPreference int value) { 232 switch (value) { 233 case OEM_NETWORK_PREFERENCE_TEST_ONLY: 234 return "OEM_NETWORK_PREFERENCE_TEST_ONLY"; 235 case OEM_NETWORK_PREFERENCE_TEST: 236 return "OEM_NETWORK_PREFERENCE_TEST"; 237 case OEM_NETWORK_PREFERENCE_UNINITIALIZED: 238 return "OEM_NETWORK_PREFERENCE_UNINITIALIZED"; 239 case OEM_NETWORK_PREFERENCE_OEM_PAID: 240 return "OEM_NETWORK_PREFERENCE_OEM_PAID"; 241 case OEM_NETWORK_PREFERENCE_OEM_PAID_NO_FALLBACK: 242 return "OEM_NETWORK_PREFERENCE_OEM_PAID_NO_FALLBACK"; 243 case OEM_NETWORK_PREFERENCE_OEM_PAID_ONLY: 244 return "OEM_NETWORK_PREFERENCE_OEM_PAID_ONLY"; 245 case OEM_NETWORK_PREFERENCE_OEM_PRIVATE_ONLY: 246 return "OEM_NETWORK_PREFERENCE_OEM_PRIVATE_ONLY"; 247 default: 248 return Integer.toHexString(value); 249 } 250 } 251 252 @Override 253 public void writeToParcel(@NonNull android.os.Parcel dest, int flags) { 254 dest.writeBundle(mNetworkMappings); 255 } 256 257 @Override 258 public int describeContents() { 259 return 0; 260 } 261 262 @NonNull 263 public static final Parcelable.Creator<OemNetworkPreferences> CREATOR = 264 new Parcelable.Creator<OemNetworkPreferences>() { 265 @Override 266 public OemNetworkPreferences[] newArray(int size) { 267 return new OemNetworkPreferences[size]; 268 } 269 270 @Override 271 public OemNetworkPreferences createFromParcel(@NonNull android.os.Parcel in) { 272 return new OemNetworkPreferences( 273 in.readBundle(getClass().getClassLoader())); 274 } 275 }; 276 } 277