forked from mgatelabs/cardboard
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdevice_params_helper.mm
166 lines (143 loc) · 6.49 KB
/
device_params_helper.mm
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
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
/*
* Copyright 2019 Google LLC
*
* 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.
*/
#import "qrcode/ios/device_params_helper.h"
#include <string>
#include <vector>
#include "qrcode/cardboard_v1/cardboard_v1.h"
#import "qrcode/ios/nsurl_session_data_handler.h"
// The value is an array with the creation time and the device params data.
static NSString *const kCardboardDeviceParamsAndTimeKey =
@"com.google.cardboard.sdk.DeviceParamsAndTime";
@implementation CardboardDeviceParamsHelper
+ (NSData *)parseURL:(NSURL *)url {
if (!url) {
return nil;
}
if ([NSURLSessionDataHandler isOriginalCardboardDeviceUrl:url]) {
return [CardboardDeviceParamsHelper createCardboardV1Params];
} else if ([NSURLSessionDataHandler isCardboardDeviceUrl:url]) {
return [CardboardDeviceParamsHelper readDeviceParamsFromUrl:url];
}
return nil;
}
+ (void)readFromUrl:(NSURL *)url
withCompletion:(void (^)(NSData *deviceParams, NSError *error))completion {
NSMutableURLRequest *request =
[NSMutableURLRequest requestWithURL:url
cachePolicy:NSURLRequestReloadIgnoringCacheData
timeoutInterval:10];
// Save bandwidth, just read the header.
[request setHTTPMethod:@"HEAD"];
NSURLSessionDataHandler *dataHandler =
[[NSURLSessionDataHandler alloc] initWithOnUrlCompletion:^(NSURL *targetUrl, NSError *error) {
if (!targetUrl) {
NSLog(@"failed to result the url = %@", url);
}
completion([CardboardDeviceParamsHelper parseURL:targetUrl], error);
}];
NSURLSessionConfiguration *config = [NSURLSessionConfiguration defaultSessionConfiguration];
NSURLSession *session = [NSURLSession sessionWithConfiguration:config
delegate:dataHandler
delegateQueue:[NSOperationQueue mainQueue]];
NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request];
[dataTask resume];
}
+ (void)resolveAndUpdateViewerProfileFromURL:(NSURL *)url
withCompletion:(void (^)(BOOL success, NSError *error))completion {
// If the scheme is http, replace it with https.
NSURLComponents *components = [NSURLComponents componentsWithURL:url resolvingAgainstBaseURL:NO];
if ([components.scheme.lowercaseString isEqualToString:@"http"]) {
components.scheme = @"https";
url = [components URL];
}
// Try to parse the url if it has the params encoded in it. This can help avoid a network call.
NSData *viewerParams = [CardboardDeviceParamsHelper parseURL:url];
if (viewerParams) {
[CardboardDeviceParamsHelper update:viewerParams];
completion(YES, nil);
} else {
[CardboardDeviceParamsHelper readFromUrl:url
withCompletion:^(NSData *deviceParams, NSError *error) {
if (deviceParams) {
[CardboardDeviceParamsHelper update:deviceParams];
}
completion(deviceParams != nil, error);
}];
}
}
+ (void)saveCardboardV1Params {
NSData *deviceParams = [CardboardDeviceParamsHelper createCardboardV1Params];
[CardboardDeviceParamsHelper update:deviceParams];
}
+ (NSData *)createCardboardV1Params {
std::vector<uint8_t> deviceParams = cardboard::qrcode::getCardboardV1DeviceParams();
return [NSData dataWithBytes:deviceParams.data() length:deviceParams.size()];
}
+ (NSData *)readDeviceParamsFromUrl:(NSURL *)url {
NSURLComponents *urlComponents = [NSURLComponents componentsWithURL:url
resolvingAgainstBaseURL:NO];
for (NSURLQueryItem *queryItem in urlComponents.queryItems) {
// Device parameters decoded data is after the p paramers in the url, for example:
// https://google.com/cardboard/cfg?p=device_param_encoded_string.
if ([queryItem.name isEqualToString:@"p"]) {
NSString *encodedDeviceParams =
[CardboardDeviceParamsHelper convertToNormalBase64String:queryItem.value];
return [[NSData alloc] initWithBase64EncodedString:encodedDeviceParams options:0];
}
}
NSLog(@"No Cardboard parameters in URL: %@", url);
return nil;
}
+ (void)update:(NSData *)deviceParams {
[CardboardDeviceParamsHelper
writeDeviceParams:std::string((char *)deviceParams.bytes, deviceParams.length)];
}
+ (NSData *)readSerializedDeviceParams {
std::string deviceParams = [CardboardDeviceParamsHelper readDeviceParams];
return [NSData dataWithBytes:deviceParams.c_str() length:deviceParams.length()];
}
// Convert the url safe base64 string into normal base64 string with padding if needed.
+ (NSString *)convertToNormalBase64String:(NSString *)original {
original = [original stringByReplacingOccurrencesOfString:@"_" withString:@"/"];
original = [original stringByReplacingOccurrencesOfString:@"-" withString:@"+"];
// Add the padding with "=" if needed.
NSUInteger paddingCount = 4 - (original.length % 4);
if (paddingCount == 1) {
original = [NSString stringWithFormat:@"%@=", original];
} else if (paddingCount == 2) {
original = [NSString stringWithFormat:@"%@==", original];
} else if (paddingCount == 3) {
original = [NSString stringWithFormat:@"%@===", original];
}
return original;
}
+ (std::string)readDeviceParams {
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
NSArray *array = [defaults arrayForKey:kCardboardDeviceParamsAndTimeKey];
NSData *serializedDeviceParams = array ? array[1] : nil;
if (serializedDeviceParams) {
return std::string((char *)serializedDeviceParams.bytes, serializedDeviceParams.length);
} else {
return std::string("");
}
}
+ (bool)writeDeviceParams:(const std::string &)device_params {
NSData *deviceParams = [NSData dataWithBytes:device_params.c_str() length:device_params.length()];
NSArray *array = @[ [NSDate date], deviceParams ];
[[NSUserDefaults standardUserDefaults] setObject:array forKey:kCardboardDeviceParamsAndTimeKey];
return true;
}
@end