All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.nimbusds.oauth2.sdk.device.DeviceAuthorizationRequest Maven / Gradle / Ivy

/*
 * oauth2-oidc-sdk
 *
 * Copyright 2012-2016, Connect2id Ltd and contributors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use
 * this file except in compliance with the License. You may obtain a copy of the
 * License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed
 * under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
 * CONDITIONS OF ANY KIND, either express or implied. See the License for the
 * specific language governing permissions and limitations under the License.
 */

package com.nimbusds.oauth2.sdk.device;


import com.nimbusds.common.contenttype.ContentType;
import com.nimbusds.oauth2.sdk.*;
import com.nimbusds.oauth2.sdk.auth.ClientAuthentication;
import com.nimbusds.oauth2.sdk.http.HTTPRequest;
import com.nimbusds.oauth2.sdk.id.ClientID;
import com.nimbusds.oauth2.sdk.util.MapUtils;
import com.nimbusds.oauth2.sdk.util.MultivaluedMapUtils;
import com.nimbusds.oauth2.sdk.util.StringUtils;
import com.nimbusds.oauth2.sdk.util.URLUtils;
import net.jcip.annotations.Immutable;

import java.net.URI;
import java.util.*;


/**
 * Device authorisation request. Used to start the authorization flow for
 * browserless and input constraint devices. Supports custom request
 * parameters.
 *
 * 

Extending classes may define additional request parameters as well as * enforce tighter requirements on the base parameters. * *

Example HTTP request: * *

 * POST /device_authorization HTTP/1.1
 * Host: server.example.com
 * Content-Type: application/x-www-form-urlencoded
 *
 * client_id=459691054427
 * 
* *

Related specifications: * *

    *
  • OAuth 2.0 Device Authorization Grant (RFC 8628) *
*/ @Immutable public class DeviceAuthorizationRequest extends AbstractOptionallyIdentifiedRequest { /** * The registered parameter names. */ private static final Set REGISTERED_PARAMETER_NAMES; static { Set p = new HashSet<>(); p.add("client_id"); p.add("scope"); REGISTERED_PARAMETER_NAMES = Collections.unmodifiableSet(p); } /** * The scope (optional). */ private final Scope scope; /** * Custom parameters. */ private final Map> customParams; /** * Builder for constructing authorisation requests. */ public static class Builder { /** * The endpoint URI (optional). */ private URI uri; /** * The client authentication (optional). */ private final ClientAuthentication clientAuth; /** * The client identifier (required if not authenticated). */ private final ClientID clientID; /** * The scope (optional). */ private Scope scope; /** * Custom parameters. */ private final Map> customParams = new HashMap<>(); /** * Creates a new devize authorization request builder. * * @param clientID The client identifier. Corresponds to the {@code client_id} * parameter. Must not be {@code null}. */ public Builder(final ClientID clientID) { if (clientID == null) throw new IllegalArgumentException("The client ID must not be null"); this.clientID = clientID; this.clientAuth = null; } /** * Creates a new device authorization request builder for an * authenticated request. * * @param clientAuth The client authentication. Must not be * {@code null}. */ public Builder(final ClientAuthentication clientAuth) { if (clientAuth == null) throw new IllegalArgumentException("The client authentication must not be null"); this.clientID = null; this.clientAuth = clientAuth; } /** * Creates a new device authorization request builder from the * specified request. * * @param request The device authorization request. Must not be * {@code null}. */ public Builder(final DeviceAuthorizationRequest request) { uri = request.getEndpointURI(); clientAuth = request.getClientAuthentication(); scope = request.scope; clientID = request.getClientID(); customParams.putAll(request.getCustomParameters()); } /** * Sets the scope. Corresponds to the optional {@code scope} * parameter. * * @param scope The scope, {@code null} if not specified. * * @return This builder. */ public Builder scope(final Scope scope) { this.scope = scope; return this; } /** * Sets a custom parameter. * * @param name The parameter name. Must not be {@code null}. * @param values The parameter values, {@code null} if not * specified. * * @return This builder. */ public Builder customParameter(final String name, final String... values) { if (values == null || values.length == 0) { customParams.remove(name); } else { customParams.put(name, Arrays.asList(values)); } return this; } /** * Sets the URI of the endpoint (HTTP or HTTPS) for which the * request is intended. * * @param uri The endpoint URI, {@code null} if not specified. * * @return This builder. */ public Builder endpointURI(final URI uri) { this.uri = uri; return this; } /** * Builds a new device authorization request. * * @return The device authorization request. */ public DeviceAuthorizationRequest build() { try { if (clientAuth == null) { return new DeviceAuthorizationRequest(uri, clientID, scope, customParams); } else { return new DeviceAuthorizationRequest(uri, clientAuth, scope, customParams); } } catch (IllegalArgumentException e) { throw new IllegalStateException(e.getMessage(), e); } } } /** * Creates a new minimal device authorization request. * * @param uri The URI of the device authorization endpoint. May be * {@code null} if the {@link #toHTTPRequest} method * will not be used. * @param clientID The client identifier. Corresponds to the * {@code client_id} parameter. Must not be * {@code null}. */ public DeviceAuthorizationRequest(final URI uri, final ClientID clientID) { this(uri, clientID, null, null); } /** * Creates a new device authorization request. * * @param uri The URI of the device authorization endpoint. May be * {@code null} if the {@link #toHTTPRequest} method * will not be used. * @param clientID The client identifier. Corresponds to the * {@code client_id} parameter. Must not be * {@code null}. * @param scope The request scope. Corresponds to the optional * {@code scope} parameter. {@code null} if not * specified. */ public DeviceAuthorizationRequest(final URI uri, final ClientID clientID, final Scope scope) { this(uri, clientID, scope, null); } /** * Creates a new device authorization request with extension and custom * parameters. * * @param uri The URI of the device authorization endpoint. * May be {@code null} if the {@link #toHTTPRequest} * method will not be used. * @param clientID The client identifier. Corresponds to the * {@code client_id} parameter. Must not be * {@code null}. * @param scope The request scope. Corresponds to the optional * {@code scope} parameter. {@code null} if not * specified. * @param customParams Custom parameters, empty map or {@code null} if * none. */ public DeviceAuthorizationRequest(final URI uri, final ClientID clientID, final Scope scope, final Map> customParams) { super(uri, clientID); if (clientID == null) throw new IllegalArgumentException("The client ID must not be null"); this.scope = scope; if (MapUtils.isNotEmpty(customParams)) { this.customParams = Collections.unmodifiableMap(customParams); } else { this.customParams = Collections.emptyMap(); } } /** * Creates a new authenticated device authorization request with * extension and custom parameters. * * @param uri The URI of the device authorization endpoint. * May be {@code null} if the {@link #toHTTPRequest} * method will not be used. * @param clientAuth The client authentication. Must not be * {@code null}. * @param scope The request scope. Corresponds to the optional * {@code scope} parameter. {@code null} if not * specified. * @param customParams Custom parameters, empty map or {@code null} if * none. */ public DeviceAuthorizationRequest(final URI uri, final ClientAuthentication clientAuth, final Scope scope, final Map> customParams) { super(uri, clientAuth); if (clientAuth == null) throw new IllegalArgumentException("The client authentication must not be null"); this.scope = scope; if (MapUtils.isNotEmpty(customParams)) { this.customParams = Collections.unmodifiableMap(customParams); } else { this.customParams = Collections.emptyMap(); } } /** * Returns the registered (standard) OAuth 2.0 device authorization * request parameter names. * * @return The registered OAuth 2.0 device authorization request * parameter names, as a unmodifiable set. */ public static Set getRegisteredParameterNames() { return REGISTERED_PARAMETER_NAMES; } /** * Gets the scope. Corresponds to the optional {@code scope} parameter. * * @return The scope, {@code null} if not specified. */ public Scope getScope() { return scope; } /** * Returns the additional custom parameters. * * @return The additional custom parameters as a unmodifiable map, * empty map if none. */ public Map> getCustomParameters() { return customParams; } /** * Returns the specified custom parameter. * * @param name The parameter name. Must not be {@code null}. * * @return The parameter value(s), {@code null} if not specified. */ public List getCustomParameter(final String name) { return customParams.get(name); } /** * Returns the matching HTTP request. * * @return The HTTP request. */ @Override public HTTPRequest toHTTPRequest() { if (getEndpointURI() == null) throw new SerializeException("The endpoint URI is not specified"); HTTPRequest httpRequest = new HTTPRequest(HTTPRequest.Method.POST, getEndpointURI()); httpRequest.setEntityContentType(ContentType.APPLICATION_URLENCODED); if (getClientAuthentication() != null) { getClientAuthentication().applyTo(httpRequest); } Map> params; try { params = new LinkedHashMap<>(httpRequest.getBodyAsFormParameters()); } catch (ParseException e) { throw new SerializeException(e.getMessage(), e); } if (scope != null && !scope.isEmpty()) { params.put("scope", Collections.singletonList(scope.toString())); } if (getClientID() != null) { params.put("client_id", Collections.singletonList(getClientID().getValue())); } if (!getCustomParameters().isEmpty()) { params.putAll(getCustomParameters()); } httpRequest.setBody(URLUtils.serializeParameters(params)); return httpRequest; } /** * Parses an device authorization request from the specified HTTP * request. * *

Example HTTP request (GET): * *

	 * POST /device_authorization HTTP/1.1
	 * Host: server.example.com
	 * Content-Type: application/x-www-form-urlencoded
	 *
	 * client_id=459691054427
	 * 
* * @param httpRequest The HTTP request. Must not be {@code null}. * * @return The device authorization request. * * @throws ParseException If the HTTP request couldn't be parsed to an * device authorization request. */ public static DeviceAuthorizationRequest parse(final HTTPRequest httpRequest) throws ParseException { // Only HTTP POST accepted URI uri = httpRequest.getURI(); httpRequest.ensureMethod(HTTPRequest.Method.POST); httpRequest.ensureEntityContentType(ContentType.APPLICATION_URLENCODED); // Parse client authentication, if any ClientAuthentication clientAuth; try { clientAuth = ClientAuthentication.parse(httpRequest); } catch (ParseException e) { throw new ParseException(e.getMessage(), OAuth2Error.INVALID_REQUEST.appendDescription(": " + e.getMessage())); } Map> params = httpRequest.getBodyAsFormParameters(); ClientID clientID; String v; if (clientAuth == null) { // Parse mandatory client ID for unauthenticated requests v = MultivaluedMapUtils.getFirstValue(params, "client_id"); if (StringUtils.isBlank(v)) { String msg = "Missing client_id parameter"; throw new ParseException(msg, OAuth2Error.INVALID_REQUEST.appendDescription(": " + msg)); } clientID = new ClientID(v); } else { clientID = null; } // Parse optional scope v = MultivaluedMapUtils.getFirstValue(params, "scope"); Scope scope = null; if (StringUtils.isNotBlank(v)) scope = Scope.parse(v); // Parse custom parameters Map> customParams = null; for (Map.Entry> p : params.entrySet()) { if (!REGISTERED_PARAMETER_NAMES.contains(p.getKey())) { // We have a custom parameter if (customParams == null) { customParams = new HashMap<>(); } customParams.put(p.getKey(), p.getValue()); } } if (clientAuth == null) { return new DeviceAuthorizationRequest(uri, clientID, scope, customParams); } else { return new DeviceAuthorizationRequest(uri, clientAuth, scope, customParams); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy