FRAPushMechanismFactory.m revision a8e01f8115b4f68ce12e939be8c523084367d9ef
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson * The contents of this file are subject to the terms of the Common Development and
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson * Distribution License (the License). You may not use this file except in compliance with the
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson * You can obtain a copy of the License at legal/CDDLv1.0.txt. See the License for the
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson * specific language governing permission and limitations under the License.
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson * When distributing Covered Software, include this CDDL Header Notice in each file and include
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson * the License file at legal/CDDLv1.0.txt. If applicable, add the following below the CDDL
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson * Header, with the fields enclosed by brackets [] replaced by your own identifying
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson * information: "Portions copyright [year] [name of copyright owner]".
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson * Copyright 2016 ForgeRock AS.
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson#import "FRAError.h"
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson#import "FRAIdentity.h"
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson#import "FRAIdentityDatabase.h"
f71f7a61dec7c9089378d14493ad564a1dedf0b5neil_a_wilson#import "FRAMechanismFactory.h"
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson#import "FRAMessageUtils.h"
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson#import "FRAMockURLProtocol.h"
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson#import "FRAPushMechanism.h"
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson#import "FRAPushMechanismFactory.h"
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson#import "FRAQRUtils.h"
a3d3ab94806056d2355afea6fe8daac41059b9fbludovicp#import "FRASerialization.h"
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson/*! QR code key for the secret. */
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson/*! QR code key for the authentication endpoint url. */
0f8553e2af5fc49a510ecfcfc93e66d06713f631ludoNSString *const AUTHENTICATION_ENDPOINT_URL_QR_KEY = @"a";
0f8553e2af5fc49a510ecfcfc93e66d06713f631ludo/*! QR code key for the regsitration endpoint url. */
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilsonNSString *const REGISTRATION_ENDPOINT_URL_QR_KEY = @"r";
27f8adec83293fb8bd3bfa37175322b0ee3bb933jvergara/*! QR code key for the message. */
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilson/*! QR code key for the background colour of the mechanism. */
7c5dba5aa2daa6307b5374b24275cf5c2b4d822dneil_a_wilsonNSString *const BACKGROUND_COLOUR_QR_KEY = @"b";
9da44d3de0a7180285a77b7e8d2426a72aca249ejvergara/*! QR code key for the registration challange. */
@interface FRAPushMechanismFactory ()
@end
if (self) {
return self;
- (FRAMechanism *) buildMechanism:(NSURL *)uri database:(FRAIdentityDatabase *)database identityModel:(FRAIdentityModel *)identityModel handler:(void (^)(BOOL, NSError *))handler error:(NSError *__autoreleasing *)error {
*error = [FRAError createError:NSLocalizedString(@"Cannot register while notifications are disabled or the device is offline", nil) code:FRAMissingDeviceId];
return nil;
NSString *regEndpoint = [self utf8StringFromData:[FRAQRUtils decodeURL:[query objectForKey:REGISTRATION_ENDPOINT_URL_QR_KEY]]];
NSString *authEndpoint = [self utf8StringFromData:[FRAQRUtils decodeURL:[query objectForKey:AUTHENTICATION_ENDPOINT_URL_QR_KEY]]];
NSString *challenge = [FRAQRUtils replaceCharactersForURLDecoding:[query objectForKey:REGISTRATION_CHALLENGE_QR_KEY]];
NSString *loadBalancer = [self utf8StringFromData:[FRAQRUtils decodeURL:[query objectForKey:REGISTRATION_LOAD_BALLANCE_KEY]]];
NSString *image = [self utf8StringFromData:[FRAQRUtils decodeURL:[query objectForKey:IMAGE_QR_KEY]]];
NSString *issuer = [self utf8StringFromData:[FRAQRUtils decodeURL:[query objectForKey:ISSUER_QR_KEY]]];
if (![self isValidSecret:secret] || ![self isValid:regEndpoint] || ![self isValid:authEndpoint] || ![self isValid:messageId] || ![self isValid:challenge] || ![self isValid:issuer]) {
return nil;
FRAPushMechanism* mechanism = [FRAPushMechanism pushMechanismWithDatabase:database identityModel:identityModel authEndpoint:authEndpoint secret:secret];
FRAIdentity *identity = [self identityWithIssuer:issuer accountName:_label identityModel:identityModel backgroundColor:backgroundColor image:image database:database error:error];
return nil;
[self registerMechanismWithEndpoint:regEndpoint secret:secret challenge:challenge messageId:messageId mechanismUid:mechanism.mechanismUID identity:identity mechanism:mechanism identityModel:identityModel loadBalancerCookieData:loadBalancer handler:handler];
return mechanism;
return [self isValid:secret] && [FRAQRUtils isBase64:[FRAQRUtils replaceCharactersForURLDecoding:secret]];
return nil;
return nil;
return nil;
return nil;
return nil;
return query;
- (FRAIdentity *)identityWithIssuer:(NSString *)issuer accountName:(NSString *)accountName identityModel:(FRAIdentityModel *)identityModel backgroundColor:(NSString *)backgroundColor image:(NSString *)image database:(FRAIdentityDatabase *)database error:(NSError *__autoreleasing *)error {
if (!identity) {
identity = [FRAIdentity identityWithDatabase:database identityModel:identityModel accountName:accountName issuer:issuer image:[NSURL URLWithString:image] backgroundColor:backgroundColor];
return nil;
return identity;
return NO;
return YES;
- (void)registerMechanismWithEndpoint:(NSString *)regEndpoint secret:(NSString *)secret challenge:(NSString *)c messageId:(NSString *)messageId mechanismUid:(NSString *)uid identity:(FRAIdentity *)identity mechanism:(FRAMechanism *)mechanism identityModel:(FRAIdentityModel *)identityModel loadBalancerCookieData:(NSString *)loadBalancerCookieData handler:(void(^)(BOOL, NSError *))handler {
error = [FRAError createError:@"Failed to contact server for registration" code:FRANetworkFailure underlyingError:error];
- (void)invokeRegistrationHandler:(void(^)(BOOL, NSError *))handler result:(BOOL)result error:(NSError *) error {
if (handler) {
@end