1 /* 2 * Copyright 2017 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 com.android.server.wifi; 18 19 import android.annotation.Nullable; 20 import android.util.ArraySet; 21 import android.util.Log; 22 23 import com.android.server.wifi.WifiConfigStore.StoreData; 24 import com.android.server.wifi.util.WifiConfigStoreEncryptionUtil; 25 import com.android.server.wifi.util.XmlUtil; 26 27 import org.xmlpull.v1.XmlPullParser; 28 import org.xmlpull.v1.XmlPullParserException; 29 import org.xmlpull.v1.XmlSerializer; 30 31 import java.io.IOException; 32 import java.util.Collections; 33 import java.util.Set; 34 35 /** 36 * Config store data for Wifi Wake. 37 */ 38 public class WakeupConfigStoreData implements StoreData { 39 private static final String TAG = "WakeupConfigStoreData"; 40 41 private static final String XML_TAG_FEATURE_STATE_SECTION = "FeatureState"; 42 private static final String XML_TAG_IS_ACTIVE = "IsActive"; 43 private static final String XML_TAG_IS_ONBOARDED = "IsOnboarded"; 44 private static final String XML_TAG_NOTIFICATIONS_SHOWN = "NotificationsShown"; 45 private static final String XML_TAG_NETWORK_SECTION = "Network"; 46 private static final String XML_TAG_SSID = "SSID"; 47 private static final String XML_TAG_SECURITY = "Security"; 48 49 private final DataSource<Boolean> mIsActiveDataSource; 50 private final DataSource<Boolean> mIsOnboardedDataSource; 51 private final DataSource<Integer> mNotificationsDataSource; 52 private final DataSource<Set<ScanResultMatchInfo>> mNetworkDataSource; 53 private boolean mHasBeenRead = false; 54 55 /** 56 * Interface defining a data source for the store data. 57 * 58 * @param <T> Type of data source 59 */ 60 public interface DataSource<T> { 61 /** 62 * Returns the data from the data source. 63 */ getData()64 T getData(); 65 66 /** 67 * Updates the data in the data source. 68 * 69 * @param data Data retrieved from the store 70 */ setData(T data)71 void setData(T data); 72 } 73 74 /** 75 * Creates the config store data with its data sources. 76 * 77 * @param isActiveDataSource Data source for isActive 78 * @param networkDataSource Data source for the locked network list 79 */ WakeupConfigStoreData( DataSource<Boolean> isActiveDataSource, DataSource<Boolean> isOnboardedDataSource, DataSource<Integer> notificationsDataSource, DataSource<Set<ScanResultMatchInfo>> networkDataSource)80 public WakeupConfigStoreData( 81 DataSource<Boolean> isActiveDataSource, 82 DataSource<Boolean> isOnboardedDataSource, 83 DataSource<Integer> notificationsDataSource, 84 DataSource<Set<ScanResultMatchInfo>> networkDataSource) { 85 mIsActiveDataSource = isActiveDataSource; 86 mIsOnboardedDataSource = isOnboardedDataSource; 87 mNotificationsDataSource = notificationsDataSource; 88 mNetworkDataSource = networkDataSource; 89 } 90 91 /** 92 * Returns whether the user store has been read. 93 */ hasBeenRead()94 public boolean hasBeenRead() { 95 return mHasBeenRead; 96 } 97 98 @Override serializeData(XmlSerializer out, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil)99 public void serializeData(XmlSerializer out, 100 @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) 101 throws XmlPullParserException, IOException { 102 writeFeatureState(out); 103 104 for (ScanResultMatchInfo scanResultMatchInfo : mNetworkDataSource.getData()) { 105 writeNetwork(out, scanResultMatchInfo); 106 } 107 } 108 109 /** 110 * Writes the current state of Wifi Wake to an XML output stream. 111 * 112 * @param out XML output stream 113 * @throws XmlPullParserException 114 * @throws IOException 115 */ writeFeatureState(XmlSerializer out)116 private void writeFeatureState(XmlSerializer out) 117 throws IOException, XmlPullParserException { 118 XmlUtil.writeNextSectionStart(out, XML_TAG_FEATURE_STATE_SECTION); 119 120 XmlUtil.writeNextValue(out, XML_TAG_IS_ACTIVE, mIsActiveDataSource.getData()); 121 XmlUtil.writeNextValue(out, XML_TAG_IS_ONBOARDED, mIsOnboardedDataSource.getData()); 122 XmlUtil.writeNextValue(out, XML_TAG_NOTIFICATIONS_SHOWN, 123 mNotificationsDataSource.getData()); 124 125 XmlUtil.writeNextSectionEnd(out, XML_TAG_FEATURE_STATE_SECTION); 126 } 127 128 /** 129 * Writes a {@link ScanResultMatchInfo} to an XML output stream. 130 * 131 * @param out XML output stream 132 * @param scanResultMatchInfo The ScanResultMatchInfo to serialize 133 * @throws XmlPullParserException 134 * @throws IOException 135 */ writeNetwork(XmlSerializer out, ScanResultMatchInfo scanResultMatchInfo)136 private void writeNetwork(XmlSerializer out, ScanResultMatchInfo scanResultMatchInfo) 137 throws XmlPullParserException, IOException { 138 XmlUtil.writeNextSectionStart(out, XML_TAG_NETWORK_SECTION); 139 140 XmlUtil.writeNextValue(out, XML_TAG_SSID, scanResultMatchInfo.networkSsid); 141 XmlUtil.writeNextValue(out, XML_TAG_SECURITY, scanResultMatchInfo.networkType); 142 143 XmlUtil.writeNextSectionEnd(out, XML_TAG_NETWORK_SECTION); 144 } 145 146 @Override deserializeData(XmlPullParser in, int outerTagDepth, @WifiConfigStore.Version int version, @Nullable WifiConfigStoreEncryptionUtil encryptionUtil)147 public void deserializeData(XmlPullParser in, int outerTagDepth, 148 @WifiConfigStore.Version int version, 149 @Nullable WifiConfigStoreEncryptionUtil encryptionUtil) 150 throws XmlPullParserException, IOException { 151 if (!mHasBeenRead) { 152 Log.d(TAG, "WifiWake user data has been read"); 153 mHasBeenRead = true; 154 } 155 // Ignore empty reads. 156 if (in == null) { 157 return; 158 } 159 160 Set<ScanResultMatchInfo> networks = new ArraySet<>(); 161 162 String[] headerName = new String[1]; 163 while (XmlUtil.gotoNextSectionOrEnd(in, headerName, outerTagDepth)) { 164 switch (headerName[0]) { 165 case XML_TAG_FEATURE_STATE_SECTION: 166 parseFeatureState(in, outerTagDepth + 1); 167 break; 168 case XML_TAG_NETWORK_SECTION: 169 networks.add(parseNetwork(in, outerTagDepth + 1)); 170 break; 171 } 172 } 173 174 mNetworkDataSource.setData(networks); 175 } 176 177 /** 178 * Parses the state of Wifi Wake from an XML input stream and sets the respective data sources. 179 * 180 * @param in XML input stream 181 * @param outerTagDepth XML tag depth of the containing section 182 * @throws IOException 183 * @throws XmlPullParserException 184 */ parseFeatureState(XmlPullParser in, int outerTagDepth)185 private void parseFeatureState(XmlPullParser in, int outerTagDepth) 186 throws IOException, XmlPullParserException { 187 boolean isActive = false; 188 boolean isOnboarded = false; 189 int notificationsShown = 0; 190 191 while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { 192 String[] valueName = new String[1]; 193 Object value = XmlUtil.readCurrentValue(in, valueName); 194 if (valueName[0] == null) { 195 throw new XmlPullParserException("Missing value name"); 196 } 197 switch (valueName[0]) { 198 case XML_TAG_IS_ACTIVE: 199 isActive = (Boolean) value; 200 break; 201 case XML_TAG_IS_ONBOARDED: 202 isOnboarded = (Boolean) value; 203 break; 204 case XML_TAG_NOTIFICATIONS_SHOWN: 205 notificationsShown = (Integer) value; 206 break; 207 default: 208 Log.w(TAG, "Unknown value found: " + valueName[0]); 209 break; 210 } 211 } 212 213 mIsActiveDataSource.setData(isActive); 214 mIsOnboardedDataSource.setData(isOnboarded); 215 mNotificationsDataSource.setData(notificationsShown); 216 } 217 218 /** 219 * Parses a {@link ScanResultMatchInfo} from an XML input stream. 220 * 221 * @param in XML input stream 222 * @param outerTagDepth XML tag depth of the containing section 223 * @return The {@link ScanResultMatchInfo} 224 * @throws IOException 225 * @throws XmlPullParserException 226 */ parseNetwork(XmlPullParser in, int outerTagDepth)227 private ScanResultMatchInfo parseNetwork(XmlPullParser in, int outerTagDepth) 228 throws IOException, XmlPullParserException { 229 ScanResultMatchInfo scanResultMatchInfo = new ScanResultMatchInfo(); 230 while (!XmlUtil.isNextSectionEnd(in, outerTagDepth)) { 231 String[] valueName = new String[1]; 232 Object value = XmlUtil.readCurrentValue(in, valueName); 233 if (valueName[0] == null) { 234 throw new XmlPullParserException("Missing value name"); 235 } 236 switch (valueName[0]) { 237 case XML_TAG_SSID: 238 scanResultMatchInfo.networkSsid = (String) value; 239 break; 240 case XML_TAG_SECURITY: 241 scanResultMatchInfo.networkType = (int) value; 242 break; 243 default: 244 Log.w(TAG, "Ignoring unknown tag under " + TAG + ": " + valueName[0]); 245 break; 246 } 247 } 248 249 return scanResultMatchInfo; 250 } 251 252 @Override resetData()253 public void resetData() { 254 mNetworkDataSource.setData(Collections.emptySet()); 255 mIsActiveDataSource.setData(false); 256 mIsOnboardedDataSource.setData(false); 257 mNotificationsDataSource.setData(0); 258 } 259 260 @Override hasNewDataToSerialize()261 public boolean hasNewDataToSerialize() { 262 // always persist. 263 return true; 264 } 265 266 @Override getName()267 public String getName() { 268 return TAG; 269 } 270 271 @Override getStoreFileId()272 public @WifiConfigStore.StoreFileId int getStoreFileId() { 273 // Shared general store. 274 return WifiConfigStore.STORE_FILE_USER_GENERAL; 275 } 276 } 277