1 /**
2  * Copyright (c) 2016, 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.wifi.hotspot2.pps;
18 
19 import android.os.Parcel;
20 import android.os.Parcelable;
21 import android.text.TextUtils;
22 import android.util.Log;
23 
24 import java.nio.charset.StandardCharsets;
25 import java.util.Arrays;
26 import java.util.Collections;
27 import java.util.HashMap;
28 import java.util.Map;
29 import java.util.Objects;
30 
31 /**
32  * Class representing HomeSP subtree in PerProviderSubscription (PPS)
33  * Management Object (MO) tree.
34  *
35  * For more info, refer to Hotspot 2.0 PPS MO defined in section 9.1 of the Hotspot 2.0
36  * Release 2 Technical Specification.
37  */
38 public final class HomeSp implements Parcelable {
39     private static final String TAG = "HomeSp";
40 
41     /**
42      * Maximum number of bytes allowed for a SSID.
43      */
44     private static final int MAX_SSID_BYTES = 32;
45 
46     /**
47      * Integer value used for indicating null value in the Parcel.
48      */
49     private static final int NULL_VALUE = -1;
50 
51     /**
52      * FQDN (Fully Qualified Domain Name) of this home service provider.
53      */
54     private String mFqdn = null;
55     /**
56      * Set the FQDN (Fully Qualified Domain Name) associated with this home service provider.
57      *
58      * @param fqdn The FQDN to set to
59      */
setFqdn(String fqdn)60     public void setFqdn(String fqdn) {
61         mFqdn = fqdn;
62     }
63     /**
64      * Get the FQDN (Fully Qualified Domain Name) associated with this home service provider.
65      *
66      * @return the FQDN associated with this home service provider
67      */
getFqdn()68     public String getFqdn() {
69         return mFqdn;
70     }
71 
72     /**
73      * Friendly name of this home service provider.
74      */
75     private String mFriendlyName = null;
76     /**
77      * Set the friendly name associated with this home service provider.
78      *
79      * @param friendlyName The friendly name to set to
80      */
setFriendlyName(String friendlyName)81     public void setFriendlyName(String friendlyName) {
82         mFriendlyName = friendlyName;
83     }
84     /**
85      * Get the friendly name associated with this home service provider.
86      *
87      * @return the friendly name associated with this home service provider
88      */
getFriendlyName()89     public String getFriendlyName() {
90         return mFriendlyName;
91     }
92 
93     /**
94      * Icon URL of this home service provider.
95      */
96     private String mIconUrl = null;
97     /**
98      * @hide
99      */
setIconUrl(String iconUrl)100     public void setIconUrl(String iconUrl) {
101         mIconUrl = iconUrl;
102     }
103     /**
104      * @hide
105      */
getIconUrl()106     public String getIconUrl() {
107         return mIconUrl;
108     }
109 
110     /**
111      * <SSID, HESSID> duple of the networks that are consider home networks.
112      *
113      * According to the Section 9.1.2 of the Hotspot 2.0 Release 2 Technical Specification,
114      * all nodes in the PSS MO are encoded using UTF-8 unless stated otherwise.  Thus, the SSID
115      * string is assumed to be encoded using UTF-8.
116      */
117     private Map<String, Long> mHomeNetworkIds = null;
118     /**
119      * @hide
120      */
setHomeNetworkIds(Map<String, Long> homeNetworkIds)121     public void setHomeNetworkIds(Map<String, Long> homeNetworkIds) {
122         mHomeNetworkIds = homeNetworkIds;
123     }
124     /**
125      * @hide
126      */
getHomeNetworkIds()127     public Map<String, Long> getHomeNetworkIds() {
128         return mHomeNetworkIds;
129     }
130 
131     /**
132      * Used for determining if this provider is a member of a given Hotspot provider.
133      * Every Organization Identifiers (OIs) in this list are required to match an OI in the
134      * the Roaming Consortium advertised by a Hotspot, in order to consider this provider
135      * as a member of that Hotspot provider (e.g. successful authentication with such Hotspot
136      * is possible).
137      *
138      * Refer to HomeSP/HomeOIList subtree in PerProviderSubscription (PPS) Management Object
139      * (MO) tree for more detail.
140      */
141     private long[] mMatchAllOis = null;
142     /**
143      * @hide
144      */
setMatchAllOis(long[] matchAllOis)145     public void setMatchAllOis(long[] matchAllOis) {
146         mMatchAllOis = matchAllOis;
147     }
148     /**
149      * @hide
150      */
getMatchAllOis()151     public long[] getMatchAllOis() {
152         return mMatchAllOis;
153     }
154 
155     /**
156      * Used for determining if this provider is a member of a given Hotspot provider.
157      * Matching of any Organization Identifiers (OIs) in this list with an OI in the
158      * Roaming Consortium advertised by a Hotspot, will consider this provider as a member
159      * of that Hotspot provider (e.g. successful authentication with such Hotspot
160      * is possible).
161      *
162      * {@link #mMatchAllOIs} will have precedence over this one, meaning this list will
163      * only be used for matching if {@link #mMatchAllOIs} is null or empty.
164      *
165      * Refer to HomeSP/HomeOIList subtree in PerProviderSubscription (PPS) Management Object
166      * (MO) tree for more detail.
167      */
168     private long[] mMatchAnyOis = null;
169     /**
170      * @hide
171      */
setMatchAnyOis(long[] matchAnyOis)172     public void setMatchAnyOis(long[] matchAnyOis) {
173         mMatchAnyOis = matchAnyOis;
174     }
175     /**
176      * @hide
177      */
getMatchAnyOis()178     public long[] getMatchAnyOis() {
179         return mMatchAnyOis;
180     }
181 
182     /**
183      * List of FQDN (Fully Qualified Domain Name) of partner providers.
184      * These providers should also be regarded as home Hotspot operators.
185      * This relationship is most likely achieved via a commercial agreement or
186      * operator merges between the providers.
187      */
188     private String[] mOtherHomePartners = null;
189     /**
190      * @hide
191      */
setOtherHomePartners(String[] otherHomePartners)192     public void setOtherHomePartners(String[] otherHomePartners) {
193         mOtherHomePartners = otherHomePartners;
194     }
195     /**
196      * @hide
197      */
getOtherHomePartners()198     public String[] getOtherHomePartners() {
199         return mOtherHomePartners;
200     }
201 
202     /**
203      * List of Organization Identifiers (OIs) identifying a roaming consortium of
204      * which this provider is a member.
205      */
206     private long[] mRoamingConsortiumOis = null;
207     /**
208      * Set the Organization Identifiers (OIs) identifying a roaming consortium of which this
209      * provider is a member.
210      *
211      * @param roamingConsortiumOis Array of roaming consortium OIs
212      */
setRoamingConsortiumOis(long[] roamingConsortiumOis)213     public void setRoamingConsortiumOis(long[] roamingConsortiumOis) {
214         mRoamingConsortiumOis = roamingConsortiumOis;
215     }
216     /**
217      * Get the Organization Identifiers (OIs) identifying a roaming consortium of which this
218      * provider is a member.
219      *
220      * @return array of roaming consortium OIs
221      */
getRoamingConsortiumOis()222     public long[] getRoamingConsortiumOis() {
223         return mRoamingConsortiumOis;
224     }
225 
226     /**
227      * Constructor for creating HomeSp with default values.
228      */
HomeSp()229     public HomeSp() {}
230 
231     /**
232      * Copy constructor.
233      *
234      * @param source The source to copy from
235      */
HomeSp(HomeSp source)236     public HomeSp(HomeSp source) {
237         if (source == null) {
238             return;
239         }
240         mFqdn = source.mFqdn;
241         mFriendlyName = source.mFriendlyName;
242         mIconUrl = source.mIconUrl;
243         if (source.mHomeNetworkIds != null) {
244             mHomeNetworkIds = Collections.unmodifiableMap(source.mHomeNetworkIds);
245         }
246         if (source.mMatchAllOis != null) {
247             mMatchAllOis = Arrays.copyOf(source.mMatchAllOis, source.mMatchAllOis.length);
248         }
249         if (source.mMatchAnyOis != null) {
250             mMatchAnyOis = Arrays.copyOf(source.mMatchAnyOis, source.mMatchAnyOis.length);
251         }
252         if (source.mOtherHomePartners != null) {
253             mOtherHomePartners = Arrays.copyOf(source.mOtherHomePartners,
254                     source.mOtherHomePartners.length);
255         }
256         if (source.mRoamingConsortiumOis != null) {
257             mRoamingConsortiumOis = Arrays.copyOf(source.mRoamingConsortiumOis,
258                     source.mRoamingConsortiumOis.length);
259         }
260     }
261 
262     @Override
describeContents()263     public int describeContents() {
264         return 0;
265     }
266 
267     @Override
writeToParcel(Parcel dest, int flags)268     public void writeToParcel(Parcel dest, int flags) {
269         dest.writeString(mFqdn);
270         dest.writeString(mFriendlyName);
271         dest.writeString(mIconUrl);
272         writeHomeNetworkIds(dest, mHomeNetworkIds);
273         dest.writeLongArray(mMatchAllOis);
274         dest.writeLongArray(mMatchAnyOis);
275         dest.writeStringArray(mOtherHomePartners);
276         dest.writeLongArray(mRoamingConsortiumOis);
277     }
278 
279     @Override
equals(Object thatObject)280     public boolean equals(Object thatObject) {
281         if (this == thatObject) {
282             return true;
283         }
284         if (!(thatObject instanceof HomeSp)) {
285             return false;
286         }
287         HomeSp that = (HomeSp) thatObject;
288 
289         return TextUtils.equals(mFqdn, that.mFqdn)
290                 && TextUtils.equals(mFriendlyName, that.mFriendlyName)
291                 && TextUtils.equals(mIconUrl, that.mIconUrl)
292                 && (mHomeNetworkIds == null ? that.mHomeNetworkIds == null
293                         : mHomeNetworkIds.equals(that.mHomeNetworkIds))
294                 && Arrays.equals(mMatchAllOis, that.mMatchAllOis)
295                 && Arrays.equals(mMatchAnyOis, that.mMatchAnyOis)
296                 && Arrays.equals(mOtherHomePartners, that.mOtherHomePartners)
297                 && Arrays.equals(mRoamingConsortiumOis, that.mRoamingConsortiumOis);
298     }
299 
300     @Override
hashCode()301     public int hashCode() {
302         return Objects.hash(mFqdn, mFriendlyName, mIconUrl,
303                 mHomeNetworkIds, Arrays.hashCode(mMatchAllOis),
304                 Arrays.hashCode(mMatchAnyOis), Arrays.hashCode(mOtherHomePartners),
305                 Arrays.hashCode(mRoamingConsortiumOis));
306     }
307 
308     /**
309      * Get a unique identifier for HomeSp. This identifier depends only on items that remain
310      * constant throughout the lifetime of a subscription.
311      *
312      * @hide
313      * @return a Unique identifier for a HomeSp object
314      */
getUniqueId()315     public int getUniqueId() {
316         return Objects.hash(mFqdn, mFriendlyName, mHomeNetworkIds, Arrays.hashCode(mMatchAllOis),
317                 Arrays.hashCode(mMatchAnyOis), Arrays.hashCode(mOtherHomePartners),
318                 Arrays.hashCode(mRoamingConsortiumOis));
319     }
320 
321 
322     @Override
toString()323     public String toString() {
324         StringBuilder builder = new StringBuilder();
325         builder.append("FQDN: ").append(mFqdn).append("\n");
326         builder.append("FriendlyName: ").append(mFriendlyName).append("\n");
327         builder.append("IconURL: ").append(mIconUrl).append("\n");
328         builder.append("HomeNetworkIDs: ").append(mHomeNetworkIds).append("\n");
329         builder.append("MatchAllOIs: ").append(mMatchAllOis).append("\n");
330         builder.append("MatchAnyOIs: ").append(mMatchAnyOis).append("\n");
331         builder.append("OtherHomePartners: ").append(mOtherHomePartners).append("\n");
332         builder.append("RoamingConsortiumOIs: ").append(mRoamingConsortiumOis).append("\n");
333         return builder.toString();
334     }
335 
336     /**
337      * Validate HomeSp data.
338      *
339      * @return true on success or false on failure
340      * @hide
341      */
validate()342     public boolean validate() {
343         if (TextUtils.isEmpty(mFqdn)) {
344             Log.d(TAG, "Missing FQDN");
345             return false;
346         }
347         if (TextUtils.isEmpty(mFriendlyName)) {
348             Log.d(TAG, "Missing friendly name");
349             return false;
350         }
351         // Verify SSIDs specified in the NetworkID
352         if (mHomeNetworkIds != null) {
353             for (Map.Entry<String, Long> entry : mHomeNetworkIds.entrySet()) {
354                 if (entry.getKey() == null ||
355                         entry.getKey().getBytes(StandardCharsets.UTF_8).length > MAX_SSID_BYTES) {
356                     Log.d(TAG, "Invalid SSID in HomeNetworkIDs");
357                     return false;
358                 }
359             }
360         }
361         return true;
362     }
363 
364     public static final @android.annotation.NonNull Creator<HomeSp> CREATOR =
365         new Creator<HomeSp>() {
366             @Override
367             public HomeSp createFromParcel(Parcel in) {
368                 HomeSp homeSp = new HomeSp();
369                 homeSp.setFqdn(in.readString());
370                 homeSp.setFriendlyName(in.readString());
371                 homeSp.setIconUrl(in.readString());
372                 homeSp.setHomeNetworkIds(readHomeNetworkIds(in));
373                 homeSp.setMatchAllOis(in.createLongArray());
374                 homeSp.setMatchAnyOis(in.createLongArray());
375                 homeSp.setOtherHomePartners(in.createStringArray());
376                 homeSp.setRoamingConsortiumOis(in.createLongArray());
377                 return homeSp;
378             }
379 
380             @Override
381             public HomeSp[] newArray(int size) {
382                 return new HomeSp[size];
383             }
384 
385             /**
386              * Helper function for reading a Home Network IDs map from a Parcel.
387              *
388              * @param in The Parcel to read from
389              * @return Map of home network IDs
390              */
391             private Map<String, Long> readHomeNetworkIds(Parcel in) {
392                 int size = in.readInt();
393                 if (size == NULL_VALUE) {
394                     return null;
395                 }
396                 Map<String, Long> networkIds = new HashMap<>(size);
397                 for (int i = 0; i < size; i++) {
398                     String key = in.readString();
399                     Long value = null;
400                     long readValue = in.readLong();
401                     if (readValue != NULL_VALUE) {
402                         value = Long.valueOf(readValue);
403                     }
404                     networkIds.put(key, value);
405                 }
406                 return networkIds;
407             }
408         };
409 
410     /**
411      * Helper function for writing Home Network IDs map to a Parcel.
412      *
413      * @param dest The Parcel to write to
414      * @param networkIds The map of home network IDs
415      */
writeHomeNetworkIds(Parcel dest, Map<String, Long> networkIds)416     private static void writeHomeNetworkIds(Parcel dest, Map<String, Long> networkIds) {
417         if (networkIds == null) {
418             dest.writeInt(NULL_VALUE);
419             return;
420         }
421         dest.writeInt(networkIds.size());
422         for (Map.Entry<String, Long> entry : networkIds.entrySet()) {
423             dest.writeString(entry.getKey());
424             if (entry.getValue() == null) {
425                 dest.writeLong(NULL_VALUE);
426             } else {
427                 dest.writeLong(entry.getValue());
428             }
429         }
430     }
431 }
432