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

com.launchkey.sdk.service.auth.V1AuthService Maven / Gradle / Ivy

/**
 * Copyright 2015 LaunchKey, Inc.  All rights reserved.
 * 

* Licensed under the MIT License. * You may not use this file except in compliance with the License. * A copy of the License is located in the "LICENSE.txt" file accompanying * this file. This file 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.launchkey.sdk.service.auth; import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import com.fasterxml.jackson.annotation.JsonProperty; import com.launchkey.sdk.cache.PingResponseCache; import com.launchkey.sdk.crypto.Crypto; import com.launchkey.sdk.service.V1ServiceAbstract; import com.launchkey.sdk.service.error.ApiException; import com.launchkey.sdk.service.error.InvalidCallbackException; import com.launchkey.sdk.service.error.InvalidResponseException; import com.launchkey.sdk.service.error.InvalidSignatureException; import com.launchkey.sdk.transport.v1.Transport; import com.launchkey.sdk.transport.v1.domain.*; import com.launchkey.sdk.transport.v1.domain.Policy.Factor; import com.launchkey.sdk.transport.v1.domain.Policy.Policy; import java.io.IOException; import java.util.*; /** * Auth service based on an API V1 transport */ public class V1AuthService extends V1ServiceAbstract implements AuthService { /** * Constructor * * @param transport Transport service * @param crypto Crypto service * @param pingResponseCache Ping Response Cache service * @param appKey Application key for requests * @param secretKey Secret key for requests */ public V1AuthService( Transport transport, Crypto crypto, PingResponseCache pingResponseCache, long appKey, String secretKey ) { super(transport, crypto, pingResponseCache, appKey, secretKey); } @Override public String authorize(String username, String context) throws ApiException { return authorize(username, context, null); } @Override public String authorize(String username, String context, AuthPolicy policy) throws ApiException { return auth(username, context, policy, false); } @Override public String authorize(String username) throws ApiException { return authorize(username, null, null); } @Override public String login(String username, String context) throws ApiException { return login(username, context, null); } @Override public String login(String username, String context, AuthPolicy policy) throws ApiException { return auth(username, context, policy, true); } @Override public String login(String username) throws ApiException { return login(username, null, null); } @Override public void logout(String authRequestId) throws ApiException { byte[] secret = getSecret(); transport.logs( new LogsRequest( "Revoke", true, authRequestId, appKey, base64.encodeAsString(secret), base64.encodeAsString(crypto.sign(secret)) ) ); } @Override public AuthResponse getAuthResponse(String authRequestId) throws ApiException { AuthResponse authResponse = null; byte[] secret = getSecret(); PollResponse response = transport.poll( new PollRequest( authRequestId, appKey, base64.encodeAsString(secret), base64.encodeAsString(crypto.sign(secret)) ) ); if (response != null) { byte[] authJSON = crypto.decryptRSA(base64.decode(response.getAuth())); try { Auth auth = objectMapper.readValue(authJSON, Auth.class); if (!authRequestId.equals(auth.getAuthRequestId())) { throw new InvalidResponseException( "Requested auth request ID " + authRequestId + " did not match responses auth request ID " + auth.getAuthRequestId(), 0 ); } byte[] logsSecret = getSecret(); transport.logs( new LogsRequest( "Authenticate", auth.isAuthorized(), authRequestId, appKey, base64.encodeAsString(logsSecret), base64.encodeAsString(crypto.sign(logsSecret)) ) ); authResponse = new AuthResponse( auth.getAuthRequestId(), auth.isAuthorized(), response.getUserHash(), response.getOrganizationUser(), response.getUserPushId(), auth.getDeviceId() ); } catch (IOException e) { throw new InvalidResponseException("Error reading auth from poll response", e); } } return authResponse; } @Override public CallbackResponse handleCallback( Map callbackData, int signatureTimeThreshold ) throws ApiException { CallbackResponse callbackResponse; if (callbackData.containsKey("auth") && callbackData.containsKey("user_hash") && callbackData.containsKey("auth_request") ) { try { byte[] authJSON = crypto.decryptRSA(base64.decode(callbackData.get("auth"))); Auth auth = objectMapper.readValue(authJSON, Auth.class); String authRequestId = callbackData.get("auth_request"); if (!authRequestId.equals(auth.getAuthRequestId())) { throw new InvalidResponseException( "Requested auth request ID " + authRequestId + " did not match responses auth request ID " + auth.getAuthRequestId(), 0 ); } byte[] logsSecret = getSecret(); transport.logs( new LogsRequest( "Authenticate", auth.isAuthorized(), authRequestId, appKey, base64.encodeAsString(logsSecret), base64.encodeAsString(crypto.sign(logsSecret)) ) ); callbackResponse = new AuthResponseCallbackResponse( new AuthResponse( auth.getAuthRequestId(), auth.isAuthorized(), callbackData.get("user_hash"), callbackData.get("organization_user"), callbackData.get("user_push_id"), auth.getDeviceId() ) ); } catch (IOException e) { throw new InvalidCallbackException("Unable to parse auth data", e); } } else if (callbackData.containsKey("deorbit") && callbackData.containsKey("signature")) { if (!crypto.verifySignature( base64.decode(callbackData.get("signature")), callbackData.get("deorbit").getBytes(), getLaunchKeyPublicKey() ) ) { throw new InvalidSignatureException("Invalid signature", 0); } try { DeOrbitCallbackResponse cbr = objectMapper.readValue( callbackData.get("deorbit"), DeOrbitCallbackResponse.class ); // Reduce current time to the second as seconds are hte precision of "LaunchKey Time" long currentTime = new Date().getTime() / 1000 * 1000; if (cbr.getDeOrbitTime().getTime() + (signatureTimeThreshold * 1000) < currentTime) { throw new InvalidCallbackException("Message has expired"); } callbackResponse = cbr; } catch (IOException e) { throw new InvalidCallbackException("Unable to parse deorbit data", e); } } else { throw new InvalidCallbackException("Callback data was insufficient to process"); } return callbackResponse; } /** * Handle a callback request * * @param callbackData Callback data received * @return parsed callback * @throws ApiException when an error occurs handling the callback */ public CallbackResponse handleCallback(Map callbackData) throws ApiException { return handleCallback(callbackData, 300); } private String auth(String username, String context, AuthPolicy authPolicy, boolean session) throws ApiException { if (context != null && context.length() > 400) { throw new IllegalArgumentException("Context value cannot be more than 400 characters"); } Policy policy; if (authPolicy == null) { policy = null; } else { List minimumRequirements = new ArrayList(); minimumRequirements.add(new Policy.MinimumRequirement( Policy.MinimumRequirement.Type.AUTHENTICATED, authPolicy.getRequiredFactors(), authPolicy.isKnowledgeFactorRequired() ? 1 : 0, authPolicy.isInherenceFactorRequired() ? 1 : 0, authPolicy.isPossessionFactorRequired() ? 1 : 0 )); List factors; if (authPolicy.getLocations().isEmpty()) { factors = new ArrayList(); } else { List locations = new ArrayList(); for (AuthPolicy.Location location : authPolicy.getLocations()) { locations.add(new Factor.Location( location.getRadius(), location.getLatitude(), location.getLongitude() )); } factors = new ArrayList(); factors.add(new Factor( Factor.Type.GEOFENCE, true, Factor.Requirement.FORCED, 1, new Factor.Attributes(locations) )); } policy = new Policy(minimumRequirements, factors); } byte[] secret = getSecret(); AuthsRequest request = new AuthsRequest( username, appKey, base64.encodeAsString(secret), base64.encodeAsString(crypto.sign(secret)), session ? 1 : 0, 1, context, policy ); AuthsResponse response = transport.auths(request); return response.getAuthRequestId(); } /** * Transport object for de-serializing V1 auth responses */ @JsonIgnoreProperties(ignoreUnknown = true) public static final class Auth { /** * Unique identifier for the authentication request generated by the LaunchKey Engine API. */ private final String authRequestId; /** * Identifier that is unique for a single user identifying the device the user utilized to respond to the * auth request. */ private final String deviceId; /** * Has the request been authorized by the user. */ private final boolean authorized; /** * @param authorized The users response to the auth request. True and False are authorized and declined * respectively. * @param authRequestId Unique identifier for the authentication request for which this is the response. * @param deviceId Identifier for the device the User utilized to respond to the authentication request. * This value is unique to the User that responded to the authentication request and may * be duplicated in another User. */ @JsonCreator public Auth( @JsonProperty("auth_request") String authRequestId, @JsonProperty("response") boolean authorized, @JsonProperty("device_id") String deviceId ) { this.authRequestId = authRequestId; this.deviceId = deviceId; this.authorized = authorized; } /** * Get the unique identifier for the authentication request for which this is the response. * * @return Authentication request ID */ public String getAuthRequestId() { return authRequestId; } /** * Get the users response to the auth request. True and False are authorized and declined respectively. * * @return Authorized response value */ public boolean isAuthorized() { return authorized; } /** * Get the identifier for the device the User utilized to respond to the authentication request. This * value is unique to the User that responded to the authentication request and may be duplicated in another User. * * @return Device ID */ public String getDeviceId() { return deviceId; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy