1 /*
2  * Copyright (C) 2018 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.hotspot2.osulogin;
18 
19 import static android.net.NetworkCapabilities.NET_CAPABILITY_TRUSTED;
20 
21 import android.app.Activity;
22 import android.content.Context;
23 import android.graphics.Bitmap;
24 import android.net.ConnectivityManager;
25 import android.net.Network;
26 import android.net.NetworkCapabilities;
27 import android.net.NetworkRequest;
28 import android.net.http.SslError;
29 import android.net.wifi.WifiManager;
30 import android.os.Bundle;
31 import android.util.Log;
32 import android.view.KeyEvent;
33 import android.view.View;
34 import android.webkit.SslErrorHandler;
35 import android.webkit.WebChromeClient;
36 import android.webkit.WebResourceError;
37 import android.webkit.WebResourceRequest;
38 import android.webkit.WebSettings;
39 import android.webkit.WebView;
40 import android.webkit.WebViewClient;
41 import android.widget.ProgressBar;
42 import android.widget.Toast;
43 
44 import androidx.annotation.Nullable;
45 import androidx.swiperefreshlayout.widget.SwipeRefreshLayout;
46 
47 import java.net.MalformedURLException;
48 import java.net.URL;
49 
50 /**
51  * Online Sign Up Login Web View launched during Provision Process of Hotspot 2.0 rel2.
52  */
53 public class OsuLoginActivity extends Activity {
54     private static final String TAG = "OsuLogin";
55     private static final boolean DBG = true;
56 
57     private String mUrl;
58     private String mHostName;
59     private Network mNetwork;
60     private ConnectivityManager mCm;
61     private ConnectivityManager.NetworkCallback mNetworkCallback;
62     private WifiManager mWifiManager;
63     private WebView mWebView;
64     private SwipeRefreshLayout mSwipeRefreshLayout;
65     private ProgressBar mProgressBar;
66     private boolean mForceDisconnect = true;
67     boolean mRedirectResponseReceived = false;
68 
69     @Override
onCreate(@ullable Bundle savedInstanceState)70     protected void onCreate(@Nullable Bundle savedInstanceState) {
71         super.onCreate(savedInstanceState);
72         if (DBG) {
73             Log.d(TAG, "onCreate: Opening OSU Web View");
74         }
75 
76         mWifiManager = (WifiManager) getApplicationContext().getSystemService(Context.WIFI_SERVICE);
77         if (mWifiManager == null) {
78             Log.e(TAG, "Cannot get wifi service");
79             finishAndRemoveTask();
80             return;
81         }
82 
83         if (getIntent() == null) {
84             Log.e(TAG, "Intent is null");
85             finishAndRemoveTask();
86             return;
87         }
88 
89         mNetwork = getIntent().getParcelableExtra(WifiManager.EXTRA_OSU_NETWORK);
90         if (mNetwork == null) {
91             Log.e(TAG, "Cannot get the network instance for OSU from intent");
92             finishAndRemoveTask();
93             return;
94         }
95 
96         mUrl = getIntent().getStringExtra(WifiManager.EXTRA_URL);
97         if (mUrl == null) {
98             Log.e(TAG, "Cannot get OSU server url from intent");
99             finishAndRemoveTask();
100             return;
101         }
102 
103         mHostName = getHost(mUrl);
104         if (mHostName == null) {
105             Log.e(TAG, "Cannot get host from the url");
106             finishAndRemoveTask();
107             return;
108         }
109 
110         mCm = (ConnectivityManager) getApplicationContext().getSystemService(
111                 Context.CONNECTIVITY_SERVICE);
112         if (mCm == null) {
113             Log.e(TAG, "Cannot get connectivity service");
114             finishAndRemoveTask();
115             return;
116         }
117 
118         if (!mCm.bindProcessToNetwork(mNetwork)) {
119             Log.e(TAG, "Network is no longer valid");
120             finishAndRemoveTask();
121             return;
122         }
123 
124         final NetworkCapabilities networkCapabilities = mCm.getNetworkCapabilities(mNetwork);
125         if (networkCapabilities == null || !networkCapabilities.hasTransport(
126                 NetworkCapabilities.TRANSPORT_WIFI)) {
127             Log.e(TAG, "WiFi is not supported for the Network");
128             finishAndRemoveTask();
129             return;
130         }
131 
132         getActionBar().setDisplayShowHomeEnabled(false);
133         getActionBar().setElevation(0); // remove shadow
134         getActionBar().setTitle(getString(R.string.action_bar_label));
135         getActionBar().setSubtitle("");
136         setContentView(R.layout.osu_web_view);
137 
138         // Exit this app if network disappeared.
139         mNetworkCallback = new ConnectivityManager.NetworkCallback() {
140             @Override
141             public void onLost(Network network) {
142                 if (DBG) {
143                     Log.d(TAG, "Lost for the current Network, close the browser");
144                 }
145                 mForceDisconnect = false; // It is already disconnected.
146                 if (!mRedirectResponseReceived) {
147                     showSignUpFailedToast();
148                 }
149                 if (mNetwork.equals(network)) {
150                     finishAndRemoveTask();
151                 }
152             }
153         };
154 
155         mCm.registerNetworkCallback(
156                 new NetworkRequest.Builder().addTransportType(
157                         NetworkCapabilities.TRANSPORT_WIFI).removeCapability(
158                         NET_CAPABILITY_TRUSTED).build(),
159                 mNetworkCallback);
160 
161         mWebView = findViewById(R.id.webview);
162         mWebView.clearCache(true);
163         WebSettings webSettings = mWebView.getSettings();
164         webSettings.setJavaScriptEnabled(true);
165         webSettings.setMixedContentMode(WebSettings.MIXED_CONTENT_COMPATIBILITY_MODE);
166         webSettings.setUseWideViewPort(true);
167         webSettings.setLoadWithOverviewMode(true);
168         webSettings.setSupportZoom(true);
169         webSettings.setBuiltInZoomControls(true);
170         webSettings.setDisplayZoomControls(false);
171         mProgressBar = findViewById(R.id.progress_bar);
172         mWebView.setWebViewClient(new OsuWebViewClient());
173         mWebView.setWebChromeClient(new WebChromeClient() {
174             @Override
175             public void onProgressChanged(WebView view, int newProgress) {
176                 mProgressBar.setProgress(newProgress);
177             }
178         });
179 
180         if (DBG) {
181             Log.d(TAG, "OSU Web View to " + mUrl);
182         }
183 
184         mWebView.loadUrl(mUrl);
185         mSwipeRefreshLayout = findViewById(R.id.swipe_refresh);
186         mSwipeRefreshLayout.setOnRefreshListener(() -> {
187             mWebView.reload();
188             mSwipeRefreshLayout.setRefreshing(true);
189         });
190     }
191 
192     @Override
onKeyDown(int keyCode, KeyEvent event)193     public boolean onKeyDown(int keyCode, KeyEvent event) {
194         // Check if the key event was the Back button.
195         if ((keyCode == KeyEvent.KEYCODE_BACK)) {
196             // If there is a history to move back
197             if (mWebView.canGoBack()) {
198                 mWebView.goBack();
199                 return true;
200             }
201         }
202         return super.onKeyDown(keyCode, event);
203     }
204 
205     @Override
onDestroy()206     protected void onDestroy() {
207         if (mNetworkCallback != null) {
208             mCm.unregisterNetworkCallback(mNetworkCallback);
209             mNetworkCallback = null;
210         }
211         if (mWifiManager != null && mForceDisconnect) {
212             mWifiManager.disconnect();
213             mWifiManager = null;
214         }
215         super.onDestroy();
216     }
217 
getHost(String url)218     private String getHost(String url) {
219         try {
220             return new URL(url).getHost();
221         } catch (MalformedURLException e) {
222             Log.e(TAG, "Invalid URL " + url);
223         }
224         return null;
225     }
226 
getHeaderSubtitle(String urlString)227     private String getHeaderSubtitle(String urlString) {
228         try {
229             URL url = new URL(urlString);
230             return url.getProtocol() + "://" +  url.getHost();
231         } catch (MalformedURLException e) {
232             Log.e(TAG, "Invalid URL " + urlString);
233         }
234         return "";
235     }
236 
showSignUpFailedToast()237     private void showSignUpFailedToast() {
238         Toast.makeText(getApplicationContext(), R.string.sign_up_failed,
239                 Toast.LENGTH_SHORT).show();
240     }
241 
242     private class OsuWebViewClient extends WebViewClient {
243         boolean mPageError = false;
244 
245         @Override
onPageStarted(WebView view, String urlString, Bitmap favicon)246         public void onPageStarted(WebView view, String urlString, Bitmap favicon) {
247             String subtitle = getHeaderSubtitle(urlString);
248             getActionBar().setSubtitle(subtitle);
249             mProgressBar.setVisibility(View.VISIBLE);
250         }
251 
252         @Override
onPageFinished(WebView view, String url)253         public void onPageFinished(WebView view, String url) {
254             mProgressBar.setVisibility(View.INVISIBLE);
255             mSwipeRefreshLayout.setRefreshing(false);
256 
257             // Do not show the page error on UI.
258             if (mPageError) {
259                 if (mRedirectResponseReceived) {
260                     // Do not disconnect current connection while provisioning is in progress.
261                     mForceDisconnect = false;
262                 }
263                 finishAndRemoveTask();
264             }
265         }
266 
267         @Override
onReceivedError(WebView view, WebResourceRequest request, WebResourceError error)268         public void onReceivedError(WebView view, WebResourceRequest request,
269                 WebResourceError error) {
270             if (request.getUrl().toString().startsWith("http://127.0.0.1")) {
271                 mRedirectResponseReceived = true;
272                 view.stopLoading();
273                 Log.d(TAG, "Redirect received");
274             }
275 
276             if (request.isForMainFrame()) {
277                 // This happens right after getting HTTP redirect response from an OSU server
278                 // since no more Http request is allowed to send to the OSU server.
279                 mPageError = true;
280                 Log.e(TAG, "onReceived Error for MainFrame: " + error.getErrorCode());
281             }
282         }
283 
284         @Override
onReceivedSslError(WebView view, SslErrorHandler handler, SslError error)285         public void onReceivedSslError(WebView view, SslErrorHandler handler, SslError error) {
286             Log.e(TAG, String.format("SSL error: %d, url: %s, certificate: %s",
287                     error.getPrimaryError(), error.getUrl(), error.getCertificate()));
288             mPageError = true;
289             handler.cancel();
290         }
291     }
292 }
293