yourkarma/JWT

View on GitHub
Sources/JWT/Coding/JWTCoding+VersionTwo.m

Summary

Maintainability
Test Coverage
//
//  JWTCoding+VersionTwo.m
//  JWT
//
//  Created by Lobanov Dmitry on 27.11.16.
//  Copyright © 2016 JWTIO. All rights reserved.
//

#import "JWTCoding+VersionTwo.h"
#import "JWTBase64Coder.h"

#import "JWTRSAlgorithm.h"

#import "JWTAlgorithmFactory.h"

#import "JWTAlgorithmDataHolder.h"

#import "JWTClaimsSetSerializer.h"
#import "JWTClaimsSetVerifier.h"

#import "JWTErrorDescription.h"

static inline void setError(NSError **error, NSError* value) {
    if (error) {
        *error = value;
    }
}

@implementation JWT (VersionTwo)
#pragma mark - Builder

+ (JWTBuilder *)encodePayload:(NSDictionary *)payload {
    return [JWTBuilder encodePayload:payload];
}

+ (JWTBuilder *)encodeClaimsSet:(JWTClaimsSet *)claimsSet {
    return [JWTBuilder encodeClaimsSet:claimsSet];
}

+ (JWTBuilder *)decodeMessage:(NSString *)message {
    return [JWTBuilder decodeMessage:message];
}
@end

@interface JWTBuilder()

@property (copy, nonatomic, readwrite) NSString *jwtMessage;
@property (copy, nonatomic, readwrite) NSDictionary *jwtPayload;
@property (copy, nonatomic, readwrite) NSDictionary *jwtHeaders;
@property (copy, nonatomic, readwrite) JWTClaimsSet *jwtClaimsSet;
@property (copy, nonatomic, readwrite) NSArray *jwtDataHolders;

@property (copy, nonatomic, readwrite) NSString *jwtSecret;
@property (copy, nonatomic, readwrite) NSData *jwtSecretData;
@property (copy, nonatomic, readwrite) NSString *jwtPrivateKeyCertificatePassphrase;
@property (copy, nonatomic, readwrite) NSError *jwtError;
@property (strong, nonatomic, readwrite) id<JWTAlgorithm> jwtAlgorithm;
@property (copy, nonatomic, readwrite) NSString *jwtAlgorithmName;
@property (copy, nonatomic, readwrite) NSNumber *jwtOptions;
@property (copy, nonatomic, readwrite) NSSet *algorithmWhitelist;

@property (copy, nonatomic, readwrite) JWTBuilder *(^message)(NSString *message);
@property (copy, nonatomic, readwrite) JWTBuilder *(^payload)(NSDictionary *payload);
@property (copy, nonatomic, readwrite) JWTBuilder *(^headers)(NSDictionary *headers);
@property (copy, nonatomic, readwrite) JWTBuilder *(^claimsSet)(JWTClaimsSet *claimsSet);
@property (copy, nonatomic, readwrite) JWTBuilder *(^secret)(NSString *secret);
@property (copy, nonatomic, readwrite) JWTBuilder *(^secretData)(NSData *secretData);
@property (copy, nonatomic, readwrite) JWTBuilder *(^privateKeyCertificatePassphrase)(NSString *privateKeyCertificatePassphrase);
@property (copy, nonatomic, readwrite) JWTBuilder *(^algorithm)(id<JWTAlgorithm>algorithm);
@property (copy, nonatomic, readwrite) JWTBuilder *(^algorithmName)(NSString *algorithmName);
@property (copy, nonatomic, readwrite) JWTBuilder *(^options)(NSNumber *options);
@property (copy, nonatomic, readwrite) JWTBuilder *(^whitelist)(NSArray *whitelist);
@property (copy, nonatomic, readwrite) JWTBuilder * (^addDataHolder)(JWTAlgorithmBaseDataHolder *dataHolder);
@property (copy, nonatomic, readwrite) JWTBuilder * (^constructDataHolder)(id<JWTAlgorithmDataHolderProtocol> (^block)(void));
@end

@implementation JWTBuilder (Setters)
- (instancetype)message:(NSString *)message {
    self.jwtMessage = message;
    return self;
}

- (instancetype)payload:(NSDictionary *)payload {
    self.jwtPayload = payload;
    return self;
}

- (instancetype)headers:(NSDictionary *)headers {
    self.jwtHeaders = headers;
    return self;
}

- (instancetype)claimSet:(JWTClaimsSet *)claimSet {
    self.jwtClaimsSet = claimSet;
    return self;
}

- (instancetype)secret:(NSString *)secret {
    self.jwtSecret = secret;
    return self;
}

- (instancetype)secretData:(NSData *)secretData {
    self.jwtSecretData = secretData;
    return self;
}

- (instancetype)privateKeyCertificatePassphrase:(NSString *)privateKeyCertificatePassphrase {
    self.jwtPrivateKeyCertificatePassphrase = privateKeyCertificatePassphrase;
    return self;
}

- (instancetype)algorithm:(id<JWTAlgorithm>)algorithm {
    self.jwtAlgorithm = algorithm;
    return self;
}

- (instancetype)algorithmName:(NSString *)algorithmName {
    self.jwtAlgorithmName = algorithmName;
    return self;
}

- (instancetype)options:(NSNumber *)options {
    self.jwtOptions = options;
    return self;
}

- (instancetype)whitelist:(NSArray *)whitelist {
    if (whitelist) {
        self.algorithmWhitelist = [NSSet setWithArray:whitelist];
    } else {
        self.algorithmWhitelist = nil;
    }
    return self;
}

- (instancetype)addDataHolder:(JWTAlgorithmBaseDataHolder *)dataHolder {
    if (dataHolder) {

    }
    return self;
}
@end

@implementation JWTBuilder

#pragma mark - Getters
- (id<JWTAlgorithm>)jwtAlgorithm {
    if (!_jwtAlgorithm) {
        _jwtAlgorithm = [JWTAlgorithmFactory algorithmByName:_jwtAlgorithmName];
    }
    return _jwtAlgorithm;
}

- (NSDictionary *)jwtPayload {
    return _jwtClaimsSet ? [JWTClaimsSetSerializer dictionaryWithClaimsSet:_jwtClaimsSet] : _jwtPayload;
}

#pragma mark - Fluent

- (void)setupFluent {
    __weak typeof(self) weakSelf = self;
    self.message = ^(NSString *message) {
        return [weakSelf message:message];
    };

    self.payload = ^(NSDictionary *payload) {
        return [weakSelf payload:payload];
    };

    self.headers = ^(NSDictionary *headers) {
        return [weakSelf headers:headers];
    };

    self.claimsSet = ^(JWTClaimsSet *claimSet) {
        return [weakSelf claimSet:claimSet];
    };

    self.secret = ^(NSString *secret) {
        return [weakSelf secret:secret];
    };

    self.secretData = ^(NSData *secretData) {
        return [weakSelf secretData:secretData];
    };

    self.privateKeyCertificatePassphrase = ^(NSString *privateKeyCertificatePassphrase) {
        return [weakSelf privateKeyCertificatePassphrase:privateKeyCertificatePassphrase];
    };

    self.algorithm = ^(id<JWTAlgorithm> algorithm) {
        return [weakSelf algorithm:algorithm];
    };

    self.algorithmName = ^(NSString *algorithmName) {
        return [weakSelf algorithmName:algorithmName];
    };

    self.options = ^(NSNumber *options) {
        return [weakSelf options:options];
    };

    self.whitelist = ^(NSArray *whitelist) {
        return [weakSelf whitelist:whitelist];
    };

    self.addDataHolder = ^(JWTAlgorithmBaseDataHolder *holder) {
        return [weakSelf addDataHolder:holder];
    };

    self.constructDataHolder = ^(id<JWTAlgorithmDataHolderProtocol> (^block)(void)) {
        if (block) {
            return [weakSelf addDataHolder:block()];
        }
        return weakSelf;
    };
}

#pragma mark - Initialization
+ (JWTBuilder *)encodePayload:(NSDictionary *)payload {
    return [[JWTBuilder alloc] init].payload(payload);
}

+ (JWTBuilder *)encodeClaimsSet:(JWTClaimsSet *)claimsSet {
    return [[JWTBuilder alloc] init].claimsSet(claimsSet);
}

+ (JWTBuilder *)decodeMessage:(NSString *)message {
    return [[JWTBuilder alloc] init].message(message);
}

- (instancetype)init {
    if (self = [super init]) {
        [self setupFluent];
    }
    return self;
}

#pragma mark - Encoding/Decoding

- (NSString *)encode {
    NSString *result = nil;
    self.jwtError = nil;
    result = [self encodeHelper];
    return result;
}

- (NSDictionary *)decode {
    NSDictionary *result = nil;
    self.jwtError = nil;
    result = [self decodeHelper];

    return result;
}

#pragma mark - Private

#pragma mark - Encode Helpers

- (NSString *)encodeHelper
{
    if (!self.jwtAlgorithm) {
        self.jwtError = [JWTErrorDescription errorWithCode:JWTUnspecifiedAlgorithmError];
        return nil;
    }

    NSDictionary *header = @{@"typ": @"JWT", @"alg": self.jwtAlgorithm.name};
    NSMutableDictionary *allHeaders = [header mutableCopy];

    if (self.jwtHeaders.allKeys.count > 0) {
        [allHeaders addEntriesFromDictionary:self.jwtHeaders];
    }

    NSString *headerSegment = [self encodeSegment:[allHeaders copy] withError:nil];

    if (!headerSegment) {
        // encode header segment error
        self.jwtError = [JWTErrorDescription errorWithCode:JWTEncodingHeaderError];
        return nil;
    }

    NSString *payloadSegment = [self encodeSegment:self.jwtPayload withError:nil];

    if (!payloadSegment) {
        // encode payment segment error
        self.jwtError = [JWTErrorDescription errorWithCode:JWTEncodingPayloadError];
        return nil;
    }

    NSString *signingInput = [@[headerSegment, payloadSegment] componentsJoinedByString:@"."];

    NSString *signedOutput;

    if ([self.jwtAlgorithm conformsToProtocol:@protocol(JWTRSAlgorithm)]) {
        id<JWTRSAlgorithm> jwtRsAlgorithm = (id <JWTRSAlgorithm>) self.jwtAlgorithm;
        jwtRsAlgorithm.privateKeyCertificatePassphrase = self.jwtPrivateKeyCertificatePassphrase;
    }
    __auto_type theAlgorithmHolder = [[JWTAlgorithmHolder alloc] initWithAlgorithm:self.jwtAlgorithm];
    if (self.jwtSecretData && [theAlgorithmHolder respondsToSelector:@selector(encodePayloadData:withSecret:)]) {
        NSData *signedOutputData = [theAlgorithmHolder encodePayloadData:[signingInput dataUsingEncoding:NSUTF8StringEncoding] withSecret:self.jwtSecretData];

        signedOutput = [JWTBase64Coder base64UrlEncodedStringWithData:signedOutputData];
    } else {
        NSData *signedOutputData = [theAlgorithmHolder encodePayload:signingInput withSecret:self.jwtSecret];
        signedOutput = [JWTBase64Coder base64UrlEncodedStringWithData:signedOutputData];
    }

    if (signedOutput) { // Make sure signing worked (e.g. we may have issues extracting the key from the PKCS12 bundle if passphrase is incorrect)
        return [@[headerSegment, payloadSegment, signedOutput] componentsJoinedByString:@"."];
    } else {
        self.jwtError = [JWTErrorDescription errorWithCode:JWTEncodingSigningError];
        return nil;
    }
}

- (NSString *)encodeSegment:(id)theSegment withError:(NSError **)error
{
    NSData *encodedSegmentData = nil;

    if (theSegment) {
        encodedSegmentData = [NSJSONSerialization dataWithJSONObject:theSegment options:0 error:error];
    }
    else {
        // error!
        NSError *generatedError = [JWTErrorDescription errorWithCode:JWTInvalidSegmentSerializationError];
        if (error) {
            *error = generatedError;
        }
        NSLog(@"%@ Could not encode segment: %@", self.class, generatedError.localizedDescription);
        return nil;
    }

    NSString *encodedSegment = nil;

    if (encodedSegmentData) {
        encodedSegment = [JWTBase64Coder base64UrlEncodedStringWithData:encodedSegmentData];
    }

    return encodedSegment;
}

#pragma mark - Decode Helpers

- (NSDictionary *)decodeHelper
{
    NSError *error = nil;
    NSDictionary *dictionary = [self decodeMessage:self.jwtMessage withSecret:self.jwtSecret withSecretData:self.jwtSecretData withError:&error withForcedAlgorithmByName:self.jwtAlgorithmName skipVerification:[self.jwtOptions boolValue] whitelist:self.algorithmWhitelist];

    if (error) {
        self.jwtError = error;
        return nil;
    }

    if (self.jwtClaimsSet) {
        BOOL claimVerified = [JWTClaimsSetVerifier verifyClaimsSet:[JWTClaimsSetSerializer claimsSetWithDictionary:dictionary[@"payload"]] withTrustedClaimsSet:self.jwtClaimsSet];
        if (claimVerified) {
            return dictionary;
        }
        else {
            self.jwtError = [JWTErrorDescription errorWithCode:JWTClaimsSetVerificationFailed];
            return nil;
        }
    }

    return dictionary;
}

- (NSDictionary *)decodeMessage:(NSString *)theMessage withSecret:(NSString *)theSecret withSecretData:(NSData *)secretData withError:(NSError *__autoreleasing *)theError withForcedAlgorithmByName:(NSString *)theAlgorithmName skipVerification:(BOOL)skipVerification {
    NSArray *parts = [theMessage componentsSeparatedByString:@"."];

    if (parts.count < 3) {
        // generate error?
        setError(theError, [JWTErrorDescription errorWithCode:JWTInvalidFormatError]);
        return nil;
    }

    NSString *headerPart = parts[0];
    NSString *payloadPart = parts[1];
    NSString *signedPart = parts[2];

    // decode headerPart
    NSError *jsonError = nil;
    NSData *headerData = [JWTBase64Coder dataWithBase64UrlEncodedString:headerPart];
    id headerJSON = [NSJSONSerialization JSONObjectWithData:headerData
                                                    options:0
                                                      error:&jsonError];
    if (jsonError) {
        setError(theError, [JWTErrorDescription errorWithCode:JWTDecodingHeaderError]);
        return nil;
    }
    NSDictionary *header = (NSDictionary *)headerJSON;
    if (!header) {
        setError(theError, [JWTErrorDescription errorWithCode:JWTNoHeaderError]);
        return nil;
    }

    if (!skipVerification) {
        // find algorithm

        //It is insecure to trust the header's value for the algorithm, since
        //the signature hasn't been verified yet, so an algorithm must be provided
        if (!theAlgorithmName) {
            setError(theError, [JWTErrorDescription errorWithCode:JWTUnspecifiedAlgorithmError]);
            return nil;
        }

        NSString *headerAlgorithmName = header[@"alg"];

        //If the algorithm in the header doesn't match what's expected, verification fails
        if (![theAlgorithmName isEqualToString:headerAlgorithmName]) {
            setError(theError, [JWTErrorDescription errorWithCode:JWTAlgorithmNameMismatchError]);
            return nil;
        }

        id<JWTAlgorithm> algorithm = [JWTAlgorithmFactory algorithmByName:theAlgorithmName];

        if (!algorithm) {
            setError(theError, [JWTErrorDescription errorWithCode:JWTUnsupportedAlgorithmError]);
            return nil;
            //    NSAssert(!algorithm, @"Can't decode segment!, %@", header);
        }

        // Verify the signed part
        NSString *signingInput = [@[headerPart, payloadPart] componentsJoinedByString:@"."];
        BOOL signatureValid = NO;

        __auto_type theAlgorithmHolder = [[JWTAlgorithmHolder alloc] initWithAlgorithm:algorithm];
        if (secretData && [theAlgorithmHolder respondsToSelector:@selector(verifySignedInput:withSignature:verificationKeyData:)]) {
            signatureValid = [theAlgorithmHolder verifySignedInput:signingInput withSignature:signedPart verificationKeyData:secretData];
        } else {
            signatureValid = [theAlgorithmHolder verifySignedInput:signingInput withSignature:signedPart verificationKey:theSecret];
        }

        if (!signatureValid) {
            setError(theError, [JWTErrorDescription errorWithCode:JWTInvalidSignatureError]);
            return nil;
        }
    }

    // and decode payload
    jsonError = nil;
    NSData *payloadData = [JWTBase64Coder dataWithBase64UrlEncodedString:payloadPart];
    id payloadJSON = [NSJSONSerialization JSONObjectWithData:payloadData
                                                     options:0
                                                       error:&jsonError];
    if (jsonError) {
        setError(theError, [JWTErrorDescription errorWithCode:JWTDecodingPayloadError]);
        return nil;
    }
    NSDictionary *payload = (NSDictionary *)payloadJSON;

    if (!payload) {
        setError(theError, [JWTErrorDescription errorWithCode:JWTNoPayloadError]);
        return nil;
    }

    NSDictionary *result = @{
                             @"header" : header,
                             @"payload" : payload
                             };

    return result;
}

- (NSDictionary *)decodeMessage:(NSString *)theMessage withSecret:(NSString *)theSecret withSecretData:(NSData *)secretData withError:(NSError *__autoreleasing *)theError withForcedAlgorithmByName:(NSString *)theAlgorithmName skipVerification:(BOOL)skipVerification whitelist:(NSSet *)theWhitelist
{
    /*
     many cases:
     1. whitelist 1, algorithm 1, match 1
     everything fine, match exists. just decode by algorithm name.
     2. whitelist 1, algorithm 0 // match not needed.
     use every algorithm and try to decode.
     3. whitelist 1, algorithm 1, match 0
     throw black list error.
     4. whitelist 0
     normal decode by algorithm.
     */
    if (theWhitelist) {
        if (!theAlgorithmName) {
            // name -> decoding
            NSMutableArray *tries = [@[] mutableCopy];
            NSMutableDictionary *result = nil;
            for (NSString *name in theWhitelist) {
                // special case for none algorithm.
                // none algorithm uses
                // maybe remove later?
                NSDictionary *try = nil;
                if ([name isEqualToString:@"none"]) {
                    try = [self decodeMessage:theMessage withSecret:nil withSecretData:nil withError:theError withForcedAlgorithmByName:name skipVerification:skipVerification];
                }
                else {
                    try = [self decodeMessage:theMessage withSecret:theSecret withSecretData:secretData withError:theError withForcedAlgorithmByName:name skipVerification:skipVerification];
                }
                if (try) {
                    result = [try mutableCopy];
                    result[@"tries"] = [tries copy];
                    setError(theError, nil);
                    break;
                }
                else {
                    if (theError && *theError) {
                        [tries addObject:*theError];
                    }
                }
            }
            return [result copy];
        }
        else {
            //If a whitelist is passed in, ensure the chosen algorithm is allowed
            if (![theWhitelist containsObject:theAlgorithmName]) {
                setError(theError, [JWTErrorDescription errorWithCode:JWTBlacklistedAlgorithmError]);
                return nil;
            }
        }
    }

    return [self decodeMessage:theMessage withSecret:theSecret withSecretData:secretData withError:theError withForcedAlgorithmByName:theAlgorithmName skipVerification:skipVerification];
}

@end