summaryrefslogtreecommitdiffstats
path: root/mobile/android/services/src/main/java/org/mozilla/gecko/background/fxa/oauth/FxAccountOAuthClient10.java
blob: 4f233695bc8a85bad76828350f5af5c3b39d9d83 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

package org.mozilla.gecko.background.fxa.oauth;

import java.net.URI;
import java.net.URISyntaxException;
import java.util.concurrent.Executor;

import org.mozilla.gecko.sync.ExtendedJSONObject;
import org.mozilla.gecko.sync.net.BaseResource;

import ch.boye.httpclientandroidlib.HttpResponse;

/**
 * Talk to an fxa-oauth-server to get "implicitly granted" OAuth tokens.
 * <p>
 * To use this client, you will need a pre-allocated fxa-oauth-server
 * "client_id" with special "implicit grant" permissions.
 * <p>
 * This client was written against the API documented at <a href="https://github.com/mozilla/fxa-oauth-server/blob/41538990df9e91158558ae5a8115194383ac3b05/docs/api.md">https://github.com/mozilla/fxa-oauth-server/blob/41538990df9e91158558ae5a8115194383ac3b05/docs/api.md</a>.
 */
public class FxAccountOAuthClient10 extends FxAccountAbstractClient {
  protected static final String LOG_TAG = FxAccountOAuthClient10.class.getSimpleName();

  protected static final String AUTHORIZATION_RESPONSE_TYPE = "token";

  protected static final String JSON_KEY_ACCESS_TOKEN = "access_token";
  protected static final String JSON_KEY_ASSERTION = "assertion";
  protected static final String JSON_KEY_CLIENT_ID = "client_id";
  protected static final String JSON_KEY_RESPONSE_TYPE = "response_type";
  protected static final String JSON_KEY_SCOPE = "scope";
  protected static final String JSON_KEY_STATE = "state";
  protected static final String JSON_KEY_TOKEN = "token";
  protected static final String JSON_KEY_TOKEN_TYPE = "token_type";

  // access_token: A string that can be used for authorized requests to service providers.
  // scope: A string of space-separated permissions that this token has. May differ from requested scopes, since user can deny permissions.
  // token_type: A string representing the token type. Currently will always be "bearer".
  protected static final String[] AUTHORIZATION_RESPONSE_REQUIRED_STRING_FIELDS = new String[] { JSON_KEY_ACCESS_TOKEN, JSON_KEY_SCOPE, JSON_KEY_TOKEN_TYPE };

  public FxAccountOAuthClient10(String serverURI, Executor executor) {
    super(serverURI, executor);
  }

  /**
   * Thin container for an authorization response.
   */
  public static class AuthorizationResponse {
    public final String access_token;
    public final String token_type;
    public final String scope;

    public AuthorizationResponse(String access_token, String token_type, String scope) {
      this.access_token = access_token;
      this.token_type = token_type;
      this.scope = scope;
    }
  }

  public void authorization(String client_id, String assertion, String state, String scope,
                            RequestDelegate<AuthorizationResponse> delegate) {
    final BaseResource resource;
    try {
      resource = new BaseResource(new URI(serverURI + "authorization"));
    } catch (URISyntaxException e) {
      invokeHandleError(delegate, e);
      return;
    }

    resource.delegate = new ResourceDelegate<AuthorizationResponse>(resource, delegate) {
      @Override
      public void handleSuccess(int status, HttpResponse response, ExtendedJSONObject body) {
        try {
          body.throwIfFieldsMissingOrMisTyped(AUTHORIZATION_RESPONSE_REQUIRED_STRING_FIELDS, String.class);
          String access_token = body.getString(JSON_KEY_ACCESS_TOKEN);
          String token_type = body.getString(JSON_KEY_TOKEN_TYPE);
          String scope = body.getString(JSON_KEY_SCOPE);
          delegate.handleSuccess(new AuthorizationResponse(access_token, token_type, scope));
          return;
        } catch (Exception e) {
          delegate.handleError(e);
          return;
        }
      }
    };

    final ExtendedJSONObject requestBody = new ExtendedJSONObject();
    requestBody.put(JSON_KEY_RESPONSE_TYPE, AUTHORIZATION_RESPONSE_TYPE);
    requestBody.put(JSON_KEY_CLIENT_ID, client_id);
    requestBody.put(JSON_KEY_ASSERTION, assertion);
    if (scope != null) {
      requestBody.put(JSON_KEY_SCOPE, scope);
    }
    if (state != null) {
      requestBody.put(JSON_KEY_STATE, state);
    }

    post(resource, requestBody, delegate);
  }

  public void deleteToken(final String token, final RequestDelegate<Void> delegate) {
    final BaseResource resource;
    try {
      resource = new BaseResource(new URI(serverURI + "destroy"));
    } catch (URISyntaxException e) {
      invokeHandleError(delegate, e);
      return;
    }

    resource.delegate = new ResourceDelegate<Void>(resource, delegate) {
      @Override
      public void handleSuccess(int status, HttpResponse response, ExtendedJSONObject body) {
        try {
          delegate.handleSuccess(null);
          return;
        } catch (Exception e) {
          delegate.handleError(e);
          return;
        }
      }
    };

    final ExtendedJSONObject requestBody = new ExtendedJSONObject();
    requestBody.put(JSON_KEY_TOKEN, token);
    post(resource, requestBody, delegate);
  }
}