Поддерживать сеанс cookie в Android - PullRequest
17 голосов
/ 14 июня 2010

Хорошо, у меня есть приложение для Android с формой, двумя EditText, спиннером и кнопкой входа в систему. Пользователь выбирает службу из счетчика, вводит имя пользователя и пароль и нажимает кнопку входа. Данные отправляются через POST, возвращается ответ, обрабатывается, запускается новый WebView, загружается html-строка, сгенерированная из ответа, и у меня есть домашняя страница того сервиса, который выбрал пользователь.

Это все хорошо. Теперь, когда пользователь нажимает на ссылку, информация для входа не может быть найдена, и страница просит пользователя войти снова. Мой сеанс входа в систему куда-то отбрасывается, и я не уверен, как передать информацию из класса, который контролирует основную часть моего приложения, в класс, который просто запускает веб-просмотр.

Обработчик onClick кнопки входа в систему:

private class FormOnClickListener implements View.OnClickListener {

    public void onClick(View v) {

        String actionURL, user, pwd, user_field, pwd_field;

        actionURL = "thePageURL";
        user_field = "username"; //this changes based on selections in a spinner
        pwd_field = "password"; //this changes based on selections in a spinner
        user = "theUserLogin";
        pwd = "theUserPassword";

        List<NameValuePair> myList = new ArrayList<NameValuePair>();
        myList.add(new BasicNameValuePair(user_field, user)); 
        myList.add(new BasicNameValuePair(pwd_field, pwd));

        HttpParams params = new BasicHttpParams();
        DefaultHttpClient client = new DefaultHttpClient(params);
        HttpPost post = new HttpPost(actionURL);
        HttpResponse response = null;
        BasicResponseHandler myHandler = new BasicResponseHandler();
        String endResult = null;

        try { post.setEntity(new UrlEncodedFormEntity(myList)); } 
        catch (UnsupportedEncodingException e) { e.printStackTrace(); } 

        try { response = client.execute(post); } 
        catch (ClientProtocolException e) { e.printStackTrace(); } 
        catch (IOException e) { e.printStackTrace(); }  

        try { endResult = myHandler.handleResponse(response); } 
        catch (HttpResponseException e) { e.printStackTrace(); } 
        catch (IOException e) { e.printStackTrace(); }

        List<Cookie> cookies = client.getCookieStore().getCookies();
        if (!cookies.isEmpty()) {
            for (int i = 0; i < cookies.size(); i++) {
                cookie = cookies.get(i);
            }
        }

       Intent myWebViewIntent = new Intent(MsidePortal.this, MyWebView.class);
       myWebViewIntent.putExtra("htmlString", endResult);
       myWebViewIntent.putExtra("actionURL", actionURL);
       startActivity(myWebViewIntent);
    }
}

А вот класс WebView, который обрабатывает отображение ответа:

public class MyWebView extends android.app.Activity {

    private class MyWebViewClient extends WebViewClient {

        @Override
        public boolean shouldOverrideUrlLoading(WebView view, String url) {
            view.loadUrl(url);
            return true;
        }
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);
        setContentView(R.layout.web);

        MyWebViewClient myClient = new MyWebViewClient();
        WebView webview = (WebView)findViewById(R.id.mainwebview);
        webview.getSettings().setBuiltInZoomControls(true); 
        webview.getSettings().setJavaScriptEnabled(true); 
        webview.setWebViewClient(myClient);

        Bundle extras = getIntent().getExtras();
        if(extras != null) 
        {
            // Get endResult
            String htmlString = extras.getString("htmlString");
            String actionURL = extras.getString("actionURL");

            Cookie sessionCookie = MsidePortal.cookie;
            CookieSyncManager.createInstance(this);
            CookieManager cookieManager = CookieManager.getInstance();
            if (sessionCookie != null) {
                cookieManager.removeSessionCookie();
                String cookieString = sessionCookie.getName()
                        + "=" + sessionCookie.getValue()
                        + "; domain=" + sessionCookie.getDomain();
                cookieManager.setCookie(actionURL, cookieString);
                CookieSyncManager.getInstance().sync();
            }  

            webview.loadDataWithBaseURL(actionURL, htmlString, "text/html", "utf-8", actionURL);}
        }
    }
}

У меня был смешанный успех при внедрении этого решения для файлов cookie. Похоже, что он работает для одной службы, в которую я вошел, которая, как я знаю, хранит куки на сервере (старая, архаичная, но она работает, и они не хотят ее менять). Служба, которую я пытаюсь сейчас, требует от пользователя сохранить куки на их локальном компьютере, и он не работает с этой настройкой.

Есть предложения?

Ответы [ 8 ]

7 голосов
/ 23 июня 2011

Вам нужно сохранить cookie от одного звонка к другому. Вместо создания нового DefaultHttpClient используйте этот компоновщик:

private Object mLock = new Object();
private CookieStore mCookie = null;
/**
 * Builds a new HttpClient with the same CookieStore than the previous one.
 * This allows to follow the http session, without keeping in memory the
 * full DefaultHttpClient.
 * @author Régis Décamps <decamps@users.sf.net>
 */
private HttpClient getHttpClient() {
        final DefaultHttpClient httpClient = new DefaultHttpClient();
        synchronized (mLock) {
                if (mCookie == null) {
                        mCookie = httpClient.getCookieStore();
                } else {
                        httpClient.setCookieStore(mCookie);
                }
        }
        return httpClient;
}

И оставьте класс Builder в качестве поля вашего приложения.

1 голос
/ 31 марта 2012

Используйте это в ссылке для входа. Activity

    List<Cookie> cookies = client.getCookieStore().getCookies();
    if (!cookies.isEmpty()) {
        for (int i = 0; i < cookies.size(); i++) {
            cookie = cookies.get(i);
        }
    }

    Cookie sessionCookie = cookie;

    if (sessionCookie != null) {
        String cookieString = sessionCookie.getName() + "="
                + sessionCookie.getValue() + "; domain="
                + sessionCookie.getDomain();
        cookieManager
                .setCookie("www.mydomain.com", cookieString);
        CookieSyncManager.getInstance().sync();
    }
1 голос
/ 27 января 2011

Вы можете хранить файлы cookie в общих настройках и загружать их по мере необходимости в других действиях.

Или попробуйте эту идею из аналогичного вопроса .

0 голосов
/ 22 сентября 2014

Разорвите идентификатор сессии, сохраненный в файлах cookie, для идентификации конкретного пользователя. Каждый язык имеет другое имя сеанса в заголовках http. Вы можете использовать какой-либо сетевой инструмент или браузер, чтобы узнать, как называется сессионный идентификатор.

И другим способом, я думаю, в Facebook и Twitter вы удалите весь код, связанный с сеансом, это сервер использует токен доступа для идентификации пользователя.

Я ясно?

0 голосов
/ 16 июля 2013

Вы использовали эту строку -

    if (sessionCookie != null) {
        cookieManager.removeSessionCookie();
    }

Чтобы каждый раз получать новые файлы cookie.

Похоже, вы столкнулись с той же проблемой, с которой я столкнулся, проверьте ссылку ниже -

removeSessionCookie () проблема с android (code.google., Com)

говорит, что removeSessionCookie() реализован в потоке, поэтому всякий раз, когда он вызывается; поток запускается и после вызова setCookie(url, cookieString); удаляет только что установленный вами cookie. Таким образом, для некоторых устройств он работает хорошо, поскольку removeSessionCookie() уже выполнен, в то время как для некоторых он удаляет cookie, и мы получаем эту проблему.

Я предлагаю вам удалить этот removeSessionCookie();, поскольку вы устанавливаете только один файл cookie, чтобы он не конфликтовал с другими файлами cookie. Ваш код будет работать без проблем.

0 голосов
/ 27 июля 2011

Когда запускается любое новое действие (поэтому я предполагаю, что это то же самое, когда вы запускаете новое веб-представление), оно эффективно запускает новую программу с нуля.Это новое действие не будет иметь доступа к данным какого-либо предыдущего действия (если только эти данные не переданы путем присоединения к цели).

2 возможных решения:

1) putExtra можно использовать толькодля передачи примитивных данных, поэтому для передачи чего-то более сложного вам нужно либо

  • a) Обернуть более сложную структуру в классе, который реализует интерфейс
    Parcelable, который можно сохранитьв дополнительном.

    b) Обернуть более сложную структуру в классе, который реализует интерфейс
    Serializable, который может храниться в дополнительном.

Любойиз этих подходов довольно сложный и трудоемкий.

2) Лично я очень предпочитаю подход, предложенный rds.Чтобы уточнить, когда rds говорит:

оставьте класс Builder в качестве поля вашего приложения.

Я думаю, что он имеет в виду расширить класс приложения.Любые сохраненные свойства доступны глобально для всех видов деятельности.В этой статье очень четко объясняется, как это сделать: http://www.screaming -penguin.com / node / 7746 Вы можете игнорировать материал об AsyncTask (хотя я уверен, что в некоторых случаях вы найдете в этом необходимость)точка) и просто сосредоточиться на части о расширении класса приложения.

0 голосов
/ 16 июня 2011

не знаю, если вам все еще нужен ответ, но опять-таки здесь появляется дополнительная информация, которая может помочьесли вы хотите синхронизировать файлы cookie

// ensure any cookies set by the dialog are saved
CookieSyncManager.getInstance().sync();

и хотите очистить куки

public static void clearCookies(Context context) {
        // Edge case: an illegal state exception is thrown if an instance of 
        // CookieSyncManager has not be created.  CookieSyncManager is normally
        // created by a WebKit view, but this might happen if you start the 
        // app, restore saved state, and click logout before running a UI 
        // dialog in a WebView -- in which case the app crashes
        @SuppressWarnings("unused")
        CookieSyncManager cookieSyncMngr = 
            CookieSyncManager.createInstance(context);
        CookieManager cookieManager = CookieManager.getInstance();
        cookieManager.removeAllCookie();
    }
0 голосов
/ 01 июня 2011

У меня похожая проблема несколько недель назад, потому что вы создаете новый DefaultHttpClient каждый раз, когда нажимаете кнопку. Попробуйте создать один DefaultHttpClient и использовать один и тот же DefaultHttpClient для каждого запроса, который вы пытаетесь отправить.это решило мою проблему

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...