1 /* ownCloud Android client application
2 * Copyright (C) 2012 Bartek Przybylski
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program. If not, see <http://www.gnu.org/licenses/>.
19 package com
.owncloud
.android
.ui
.activity
;
21 import java
.net
.MalformedURLException
;
24 import com
.owncloud
.android
.AccountUtils
;
25 import com
.owncloud
.android
.authenticator
.AccountAuthenticator
;
26 import com
.owncloud
.android
.authenticator
.AuthenticationRunnable
;
27 import com
.owncloud
.android
.authenticator
.OnAuthenticationResultListener
;
28 import com
.owncloud
.android
.authenticator
.OnConnectCheckListener
;
29 import com
.owncloud
.android
.ui
.dialog
.SslValidatorDialog
;
30 import com
.owncloud
.android
.ui
.dialog
.SslValidatorDialog
.OnSslValidatorListener
;
31 import com
.owncloud
.android
.network
.OwnCloudClientUtils
;
32 import com
.owncloud
.android
.operations
.ConnectionCheckOperation
;
33 import com
.owncloud
.android
.operations
.OnRemoteOperationListener
;
34 import com
.owncloud
.android
.operations
.RemoteOperation
;
35 import com
.owncloud
.android
.operations
.RemoteOperationResult
;
37 import android
.accounts
.Account
;
38 import android
.accounts
.AccountAuthenticatorActivity
;
39 import android
.accounts
.AccountManager
;
40 import android
.app
.AlertDialog
;
41 import android
.app
.Dialog
;
42 import android
.app
.ProgressDialog
;
43 import android
.content
.ContentResolver
;
44 import android
.content
.DialogInterface
;
45 import android
.content
.Intent
;
46 import android
.content
.SharedPreferences
;
47 import android
.net
.Uri
;
48 import android
.os
.Bundle
;
49 import android
.os
.Handler
;
50 import android
.preference
.PreferenceManager
;
51 import android
.text
.InputType
;
52 import android
.util
.Log
;
53 import android
.view
.View
;
54 import android
.view
.View
.OnClickListener
;
55 import android
.view
.View
.OnFocusChangeListener
;
56 import android
.view
.Window
;
57 import android
.widget
.ImageView
;
58 import android
.widget
.TextView
;
59 import com
.owncloud
.android
.R
;
61 import eu
.alefzero
.webdav
.WebdavClient
;
64 * This Activity is used to add an ownCloud account to the App
66 * @author Bartek Przybylski
69 public class AuthenticatorActivity
extends AccountAuthenticatorActivity
70 implements OnAuthenticationResultListener
, OnConnectCheckListener
, OnRemoteOperationListener
, OnSslValidatorListener
,
71 OnFocusChangeListener
, OnClickListener
{
73 private static final int DIALOG_LOGIN_PROGRESS
= 0;
74 private static final int DIALOG_SSL_VALIDATOR
= 1;
75 private static final int DIALOG_CERT_NOT_SAVED
= 2;
77 private static final String TAG
= "AuthActivity";
79 private Thread mAuthThread
;
80 private AuthenticationRunnable mAuthRunnable
;
81 //private ConnectionCheckerRunnable mConnChkRunnable = null;
82 private ConnectionCheckOperation mConnChkRunnable
;
83 private final Handler mHandler
= new Handler();
84 private String mBaseUrl
;
86 private static final String STATUS_TEXT
= "STATUS_TEXT";
87 private static final String STATUS_ICON
= "STATUS_ICON";
88 private static final String STATUS_CORRECT
= "STATUS_CORRECT";
89 private static final String IS_SSL_CONN
= "IS_SSL_CONN";
90 private int mStatusText
, mStatusIcon
;
91 private boolean mStatusCorrect
, mIsSslConn
;
92 private RemoteOperationResult mLastSslUntrustedServerResult
;
94 public static final String PARAM_USERNAME
= "param_Username";
95 public static final String PARAM_HOSTNAME
= "param_Hostname";
98 protected void onCreate(Bundle savedInstanceState
) {
99 super.onCreate(savedInstanceState
);
100 getWindow().requestFeature(Window
.FEATURE_NO_TITLE
);
101 setContentView(R
.layout
.account_setup
);
102 ImageView iv
= (ImageView
) findViewById(R
.id
.refreshButton
);
103 ImageView iv2
= (ImageView
) findViewById(R
.id
.viewPassword
);
104 TextView tv
= (TextView
) findViewById(R
.id
.host_URL
);
105 TextView tv2
= (TextView
) findViewById(R
.id
.account_password
);
107 if (savedInstanceState
!= null
) {
108 mStatusIcon
= savedInstanceState
.getInt(STATUS_ICON
);
109 mStatusText
= savedInstanceState
.getInt(STATUS_TEXT
);
110 mStatusCorrect
= savedInstanceState
.getBoolean(STATUS_CORRECT
);
111 mIsSslConn
= savedInstanceState
.getBoolean(IS_SSL_CONN
);
112 setResultIconAndText(mStatusIcon
, mStatusText
);
113 findViewById(R
.id
.buttonOK
).setEnabled(mStatusCorrect
);
115 iv
.setVisibility(View
.VISIBLE
);
117 iv
.setVisibility(View
.INVISIBLE
);
120 mStatusText
= mStatusIcon
= 0;
121 mStatusCorrect
= false
;
124 iv
.setOnClickListener(this);
125 iv2
.setOnClickListener(this);
126 tv
.setOnFocusChangeListener(this);
127 tv2
.setOnFocusChangeListener(this);
131 protected void onSaveInstanceState(Bundle outState
) {
132 outState
.putInt(STATUS_ICON
, mStatusIcon
);
133 outState
.putInt(STATUS_TEXT
, mStatusText
);
134 outState
.putBoolean(STATUS_CORRECT
, mStatusCorrect
);
135 super.onSaveInstanceState(outState
);
139 protected Dialog
onCreateDialog(int id
) {
140 Dialog dialog
= null
;
142 case DIALOG_LOGIN_PROGRESS
: {
143 ProgressDialog working_dialog
= new ProgressDialog(this);
144 working_dialog
.setMessage(getResources().getString(
145 R
.string
.auth_trying_to_login
));
146 working_dialog
.setIndeterminate(true
);
147 working_dialog
.setCancelable(true
);
149 .setOnCancelListener(new DialogInterface
.OnCancelListener() {
151 public void onCancel(DialogInterface dialog
) {
152 Log
.i(TAG
, "Login canceled");
153 if (mAuthThread
!= null
) {
154 mAuthThread
.interrupt();
159 dialog
= working_dialog
;
162 case DIALOG_SSL_VALIDATOR
: {
163 dialog
= SslValidatorDialog
.newInstance(this, mLastSslUntrustedServerResult
, this);
166 case DIALOG_CERT_NOT_SAVED
: {
167 AlertDialog
.Builder builder
= new AlertDialog
.Builder(this);
168 builder
.setMessage(getResources().getString(R
.string
.ssl_validator_not_saved
));
169 builder
.setCancelable(false
);
170 builder
.setPositiveButton(R
.string
.common_ok
, new DialogInterface
.OnClickListener() {
172 public void onClick(DialogInterface dialog
, int which
) {
176 dialog
= builder
.create();
180 Log
.e(TAG
, "Incorrect dialog called with id = " + id
);
186 protected void onPrepareDialog(int id
, Dialog dialog
, Bundle args
) {
188 case DIALOG_LOGIN_PROGRESS
:
189 case DIALOG_CERT_NOT_SAVED
:
191 case DIALOG_SSL_VALIDATOR
: {
192 ((SslValidatorDialog
)dialog
).updateResult(mLastSslUntrustedServerResult
);
196 Log
.e(TAG
, "Incorrect dialog called with id = " + id
);
200 public void onAuthenticationResult(boolean success
, String message
) {
202 TextView username_text
= (TextView
) findViewById(R
.id
.account_username
), password_text
= (TextView
) findViewById(R
.id
.account_password
);
206 url
= new URL(message
);
207 } catch (MalformedURLException e
) {
208 // should never happen
209 Log
.e(getClass().getName(), "Malformed URL: " + message
);
213 String username
= username_text
.getText().toString().trim();
214 String accountName
= username
+ "@" + url
.getHost();
215 if (url
.getPort() >= 0) {
216 accountName
+= ":" + url
.getPort();
218 Account account
= new Account(accountName
,
219 AccountAuthenticator
.ACCOUNT_TYPE
);
220 AccountManager accManager
= AccountManager
.get(this);
221 accManager
.addAccountExplicitly(account
, password_text
.getText()
224 // Add this account as default in the preferences, if there is none
226 Account defaultAccount
= AccountUtils
227 .getCurrentOwnCloudAccount(this);
228 if (defaultAccount
== null
) {
229 SharedPreferences
.Editor editor
= PreferenceManager
230 .getDefaultSharedPreferences(this).edit();
231 editor
.putString("select_oc_account", accountName
);
235 final Intent intent
= new Intent();
236 intent
.putExtra(AccountManager
.KEY_ACCOUNT_TYPE
,
237 AccountAuthenticator
.ACCOUNT_TYPE
);
238 intent
.putExtra(AccountManager
.KEY_ACCOUNT_NAME
, account
.name
);
239 intent
.putExtra(AccountManager
.KEY_AUTHTOKEN
,
240 AccountAuthenticator
.ACCOUNT_TYPE
);
241 intent
.putExtra(AccountManager
.KEY_USERDATA
, username
);
243 accManager
.setUserData(account
,
244 AccountAuthenticator
.KEY_OC_VERSION
, mConnChkRunnable
245 .getDiscoveredVersion().toString());
247 accManager
.setUserData(account
,
248 AccountAuthenticator
.KEY_OC_BASE_URL
, mBaseUrl
);
250 setAccountAuthenticatorResult(intent
.getExtras());
251 setResult(RESULT_OK
, intent
);
252 Bundle bundle
= new Bundle();
253 bundle
.putBoolean(ContentResolver
.SYNC_EXTRAS_MANUAL
, true
);
254 //getContentResolver().startSync(ProviderTableMeta.CONTENT_URI,
256 ContentResolver
.requestSync(account
, "org.owncloud", bundle
);
260 * (mConnChkRunnable.getDiscoveredVersion().compareTo(OwnCloudVersion
261 * .owncloud_v2) >= 0) { Intent i = new Intent(this,
262 * ExtensionsAvailableActivity.class); startActivity(i); }
268 dismissDialog(DIALOG_LOGIN_PROGRESS
);
269 } catch (IllegalArgumentException e
) {
270 // NOTHING TO DO ; can't find out what situation that leads to the exception in this code, but user logs signal that it happens
272 TextView tv
= (TextView
) findViewById(R
.id
.account_username
);
273 tv
.setError(message
);
276 public void onCancelClick(View view
) {
277 setResult(RESULT_CANCELED
);
281 public void onOkClick(View view
) {
283 String url
= ((TextView
) findViewById(R
.id
.host_URL
)).getText()
290 if (url
.toLowerCase().startsWith("http://")
291 || url
.toLowerCase().startsWith("https://")) {
294 continueConnection(prefix
);
297 public void onRegisterClick(View view
) {
298 Intent register
= new Intent(Intent
.ACTION_VIEW
, Uri
.parse("https://owncloud.com/mobile/new"));
299 setResult(RESULT_CANCELED
);
300 startActivity(register
);
303 private void continueConnection(String prefix
) {
304 String url
= ((TextView
) findViewById(R
.id
.host_URL
)).getText()
306 String username
= ((TextView
) findViewById(R
.id
.account_username
))
307 .getText().toString();
308 String password
= ((TextView
) findViewById(R
.id
.account_password
))
309 .getText().toString();
310 if (url
.endsWith("/"))
311 url
= url
.substring(0, url
.length() - 1);
314 String webdav_path
= AccountUtils
.getWebdavPath(mConnChkRunnable
315 .getDiscoveredVersion());
317 if (webdav_path
== null
) {
318 onAuthenticationResult(false
, getString(R
.string
.auth_bad_oc_version_title
));
323 mBaseUrl
= prefix
+ url
;
324 String url_str
= prefix
+ url
+ webdav_path
;
325 uri
= new URL(url_str
);
326 } catch (MalformedURLException e
) {
327 // should never happen
328 onAuthenticationResult(false
, getString(R
.string
.auth_incorrect_address_title
));
332 showDialog(DIALOG_LOGIN_PROGRESS
);
333 mAuthRunnable
= new AuthenticationRunnable(uri
, username
, password
, this);
334 mAuthRunnable
.setOnAuthenticationResultListener(this, mHandler
);
335 mAuthThread
= new Thread(mAuthRunnable
);
340 public void onConnectionCheckResult(ResultType type
) {
341 mStatusText
= mStatusIcon
= 0;
342 mStatusCorrect
= false
;
343 String t_url
= ((TextView
) findViewById(R
.id
.host_URL
)).getText()
344 .toString().trim().toLowerCase();
349 mStatusIcon
= android
.R
.drawable
.ic_secure
;
350 mStatusText
= R
.string
.auth_secure_connection
;
351 mStatusCorrect
= true
;
355 mStatusCorrect
= true
;
356 if (t_url
.startsWith("http://") ) {
357 mStatusText
= R
.string
.auth_connection_established
;
358 mStatusIcon
= R
.drawable
.ic_ok
;
360 mStatusText
= R
.string
.auth_nossl_plain_ok_title
;
361 mStatusIcon
= android
.R
.drawable
.ic_partial_secure
;
365 mStatusIcon
= R
.drawable
.common_error
;
366 mStatusText
= R
.string
.auth_bad_oc_version_title
;
368 case WRONG_CONNECTION
:
369 mStatusIcon
= R
.drawable
.common_error
;
370 mStatusText
= R
.string
.auth_wrong_connection_title
;
373 mStatusIcon
= R
.drawable
.common_error
;
374 mStatusText
= R
.string
.auth_timeout_title
;
376 case INCORRECT_ADDRESS
:
377 mStatusIcon
= R
.drawable
.common_error
;
378 mStatusText
= R
.string
.auth_incorrect_address_title
;
380 case SSL_UNVERIFIED_SERVER
:
381 mStatusIcon
= R
.drawable
.common_error
;
382 mStatusText
= R
.string
.auth_ssl_unverified_server_title
;
385 mStatusIcon
= R
.drawable
.common_error
;
386 mStatusText
= R
.string
.auth_ssl_general_error_title
;
388 case HOST_NOT_AVAILABLE
:
389 mStatusIcon
= R
.drawable
.common_error
;
390 mStatusText
= R
.string
.auth_unknown_host_title
;
392 case NO_NETWORK_CONNECTION
:
393 mStatusIcon
= R
.drawable
.no_network
;
394 mStatusText
= R
.string
.auth_no_net_conn_title
;
396 case INSTANCE_NOT_CONFIGURED
:
397 mStatusIcon
= R
.drawable
.common_error
;
398 mStatusText
= R
.string
.auth_not_configured_title
;
401 mStatusIcon
= R
.drawable
.common_error
;
402 mStatusText
= R
.string
.auth_unknown_error_title
;
405 mStatusIcon
= R
.drawable
.common_error
;
406 mStatusText
= R
.string
.auth_incorrect_path_title
;
409 Log
.e(TAG
, "Incorrect connection checker result type: " + type
);
411 setResultIconAndText(mStatusIcon
, mStatusText
);
413 findViewById(R
.id
.refreshButton
).setVisibility(View
.VISIBLE
);
415 findViewById(R
.id
.refreshButton
).setVisibility(View
.INVISIBLE
);
416 findViewById(R
.id
.buttonOK
).setEnabled(mStatusCorrect
);
420 public void onFocusChange(View view
, boolean hasFocus
) {
421 if (view
.getId() == R
.id
.host_URL
) {
423 TextView tv
= ((TextView
) findViewById(R
.id
.host_URL
));
424 String uri
= tv
.getText().toString().trim();
425 if (uri
.length() != 0) {
426 setResultIconAndText(R
.drawable
.progress_small
,
427 R
.string
.auth_testing_connection
);
428 //mConnChkRunnable = new ConnectionCheckerRunnable(uri, this);
429 mConnChkRunnable
= new ConnectionCheckOperation(uri
, this);
430 //mConnChkRunnable.setListener(this, mHandler);
431 //mAuthThread = new Thread(mConnChkRunnable);
432 //mAuthThread.start();
433 WebdavClient client
= OwnCloudClientUtils
.createOwnCloudClient(Uri
.parse(uri
), this);
434 mAuthThread
= mConnChkRunnable
.execute(client
, this, mHandler
);
436 findViewById(R
.id
.refreshButton
).setVisibility(
438 setResultIconAndText(0, 0);
441 // avoids that the 'connect' button can be clicked if the test was previously passed
442 findViewById(R
.id
.buttonOK
).setEnabled(false
);
444 } else if (view
.getId() == R
.id
.account_password
) {
445 ImageView iv
= (ImageView
) findViewById(R
.id
.viewPassword
);
447 iv
.setVisibility(View
.VISIBLE
);
449 TextView v
= (TextView
) findViewById(R
.id
.account_password
);
450 int input_type
= InputType
.TYPE_CLASS_TEXT
451 | InputType
.TYPE_TEXT_VARIATION_PASSWORD
;
452 v
.setInputType(input_type
);
453 iv
.setVisibility(View
.INVISIBLE
);
458 private void setResultIconAndText(int drawable_id
, int text_id
) {
459 ImageView iv
= (ImageView
) findViewById(R
.id
.action_indicator
);
460 TextView tv
= (TextView
) findViewById(R
.id
.status_text
);
462 if (drawable_id
== 0 && text_id
== 0) {
463 iv
.setVisibility(View
.INVISIBLE
);
464 tv
.setVisibility(View
.INVISIBLE
);
466 iv
.setImageResource(drawable_id
);
468 iv
.setVisibility(View
.VISIBLE
);
469 tv
.setVisibility(View
.VISIBLE
);
474 public void onClick(View v
) {
475 if (v
.getId() == R
.id
.refreshButton
) {
476 onFocusChange(findViewById(R
.id
.host_URL
), false
);
477 } else if (v
.getId() == R
.id
.viewPassword
) {
478 TextView view
= (TextView
) findViewById(R
.id
.account_password
);
479 int input_type
= view
.getInputType();
480 if ((input_type
& InputType
.TYPE_TEXT_VARIATION_VISIBLE_PASSWORD
) == InputType
.TYPE_TEXT_VARIATION_VISIBLE_PASSWORD
) {
481 input_type
= InputType
.TYPE_CLASS_TEXT
482 | InputType
.TYPE_TEXT_VARIATION_PASSWORD
;
484 input_type
= InputType
.TYPE_CLASS_TEXT
485 | InputType
.TYPE_TEXT_VARIATION_VISIBLE_PASSWORD
;
487 view
.setInputType(input_type
);
492 public void onRemoteOperationFinish(RemoteOperation operation
, RemoteOperationResult result
) {
493 if (operation
.equals(mConnChkRunnable
)) {
495 mStatusText
= mStatusIcon
= 0;
496 mStatusCorrect
= false
;
497 String t_url
= ((TextView
) findViewById(R
.id
.host_URL
)).getText()
498 .toString().trim().toLowerCase();
500 switch (result
.getCode()) {
503 mStatusIcon
= android
.R
.drawable
.ic_secure
;
504 mStatusText
= R
.string
.auth_secure_connection
;
505 mStatusCorrect
= true
;
511 mStatusCorrect
= true
;
512 if (t_url
.startsWith("http://") ) {
513 mStatusText
= R
.string
.auth_connection_established
;
514 mStatusIcon
= R
.drawable
.ic_ok
;
516 mStatusText
= R
.string
.auth_nossl_plain_ok_title
;
517 mStatusIcon
= android
.R
.drawable
.ic_partial_secure
;
523 mStatusIcon
= R
.drawable
.common_error
;
524 mStatusText
= R
.string
.auth_bad_oc_version_title
;
526 case WRONG_CONNECTION
:
527 mStatusIcon
= R
.drawable
.common_error
;
528 mStatusText
= R
.string
.auth_wrong_connection_title
;
531 mStatusIcon
= R
.drawable
.common_error
;
532 mStatusText
= R
.string
.auth_timeout_title
;
534 case INCORRECT_ADDRESS
:
535 mStatusIcon
= R
.drawable
.common_error
;
536 mStatusText
= R
.string
.auth_incorrect_address_title
;
539 case SSL_RECOVERABLE_PEER_UNVERIFIED
:
540 mStatusIcon
= R
.drawable
.common_error
;
541 mStatusText
= R
.string
.auth_ssl_unverified_server_title
;
542 mLastSslUntrustedServerResult
= result
;
543 showDialog(DIALOG_SSL_VALIDATOR
);
547 mStatusIcon
= R
.drawable
.common_error
;
548 mStatusText
= R
.string
.auth_ssl_general_error_title
;
551 case HOST_NOT_AVAILABLE
:
552 mStatusIcon
= R
.drawable
.common_error
;
553 mStatusText
= R
.string
.auth_unknown_host_title
;
555 case NO_NETWORK_CONNECTION
:
556 mStatusIcon
= R
.drawable
.no_network
;
557 mStatusText
= R
.string
.auth_no_net_conn_title
;
559 case INSTANCE_NOT_CONFIGURED
:
560 mStatusIcon
= R
.drawable
.common_error
;
561 mStatusText
= R
.string
.auth_not_configured_title
;
564 mStatusIcon
= R
.drawable
.common_error
;
565 mStatusText
= R
.string
.auth_incorrect_path_title
;
567 case UNHANDLED_HTTP_CODE
:
569 mStatusIcon
= R
.drawable
.common_error
;
570 mStatusText
= R
.string
.auth_unknown_error_title
;
573 Log
.e(TAG
, "Incorrect connection checker result type: " + result
.getHttpCode());
575 setResultIconAndText(mStatusIcon
, mStatusText
);
577 findViewById(R
.id
.refreshButton
).setVisibility(View
.VISIBLE
);
579 findViewById(R
.id
.refreshButton
).setVisibility(View
.INVISIBLE
);
580 findViewById(R
.id
.buttonOK
).setEnabled(mStatusCorrect
);
585 public void onSavedCertificate() {
586 mAuthThread
= mConnChkRunnable
.retry(this, mHandler
);
590 public void onFailedSavingCertificate() {
591 showDialog(DIALOG_CERT_NOT_SAVED
);