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
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
|
/*
* cocos2d for iPhone: http://www.cocos2d-iphone.org
*
* Copyright (c) 2008-2010 Ricardo Quesada
* Copyright (c) 2011 Zynga Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*
*/
#import "CLScoreServerPost.h"
#import "ccMacros.h"
// free function used to sort
NSInteger alphabeticSort(id string1, id string2, void *reverse);
NSInteger alphabeticSort(id string1, id string2, void *reverse)
{
if ((NSInteger *)reverse == NO)
return [string2 localizedCaseInsensitiveCompare:string1];
return [string1 localizedCaseInsensitiveCompare:string2];
}
@interface CLScoreServerPost (Private)
-(void) addValue:(NSString*)value key:(NSString*)key;
-(void) calculateHashAndAddValue:(id)value key:(NSString*)key;
-(NSString*) getHashForData;
-(NSData*) getBodyValues;
-(NSString*) encodeData:(NSString*)data;
-(NSMutableURLRequest *) scoreServerRequestWithURLString:(NSString *)url;
-(BOOL) submitScore:(NSDictionary*)dict forUpdate:(BOOL)isUpdate;
@end
@implementation CLScoreServerPost
@synthesize postStatus = postStatus_;
@synthesize ranking = ranking_;
@synthesize scoreDidUpdate = scoreDidUpdate_;
@synthesize connection = connection_;
+(id) serverWithGameName:(NSString*) name gameKey:(NSString*) key delegate:(id) delegate
{
return [[[self alloc] initWithGameName:name gameKey:key delegate:delegate] autorelease];
}
-(id) initWithGameName:(NSString*) name gameKey:(NSString*) key delegate:(id)aDelegate
{
self = [super init];
if( self ) {
gameKey = [key retain];
gameName = [name retain];
bodyValues = [[NSMutableArray arrayWithCapacity:5] retain];
delegate = [aDelegate retain];
receivedData = [[NSMutableData data] retain];
ranking_ = kServerPostInvalidRanking;
}
return self;
}
-(void) dealloc
{
CCLOGINFO(@"deallocing %@", self);
[delegate release];
[gameKey release];
[gameName release];
[bodyValues release];
[receivedData release];
[connection_ release];
[super dealloc];
}
#pragma mark ScoreServer send scores
-(BOOL) sendScore: (NSDictionary*) dict
{
return [self submitScore:dict forUpdate:NO];
}
-(BOOL) updateScore: (NSDictionary*) dict
{
if (![dict objectForKey:@"cc_playername"]) {
// fail. cc_playername + cc_device_id are needed to update an score
[NSException raise:@"cocosLive:updateScore" format:@"cc_playername not found"];
}
return [self submitScore:dict forUpdate:YES];
}
-(BOOL) submitScore: (NSDictionary*)dict forUpdate:(BOOL)isUpdate
{
[receivedData setLength:0];
[bodyValues removeAllObjects];
// reset status
postStatus_ = kPostStatusOK;
// create the request
NSMutableURLRequest *post = [self scoreServerRequestWithURLString:(isUpdate ? SCORE_SERVER_UPDATE_URL : SCORE_SERVER_SEND_URL)];
CC_MD5_Init( &md5Ctx);
// hash SHALL be calculated in certain order
NSArray *keys = [dict allKeys];
int reverseSort = NO;
NSArray *sortedKeys = [keys sortedArrayUsingFunction:alphabeticSort context:&reverseSort];
for( id key in sortedKeys )
[self calculateHashAndAddValue:[dict objectForKey:key] key:key];
// device id is hashed to prevent spoofing this same score from different devices
// one way to prevent a replay attack is to send cc_id & cc_time and use it as primary keys
[self addValue:[[UIDevice currentDevice] uniqueIdentifier] key:@"cc_device_id"];
[self addValue:gameName key:@"cc_gamename"];
[self addValue:[self getHashForData] key:@"cc_hash"];
[self addValue:SCORE_SERVER_PROTOCOL_VERSION key:@"cc_prot_ver"];
[post setHTTPBody: [self getBodyValues] ];
// create the connection with the request
// and start loading the data
self.connection=[NSURLConnection connectionWithRequest:post delegate:self];
if ( ! connection_)
return NO;
return YES;
}
-(NSMutableURLRequest *) scoreServerRequestWithURLString:(NSString *)url {
NSMutableURLRequest *request=[NSMutableURLRequest requestWithURL:[NSURL URLWithString: url]
cachePolicy:NSURLRequestUseProtocolCachePolicy
timeoutInterval:10.0];
[request setHTTPMethod: @"POST"];
[request setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"];
return request;
}
-(void) calculateHashAndAddValue:(id) value key:(NSString*) key
{
NSString *val;
// value shall be a string or nsnumber
if( [value respondsToSelector:@selector(stringValue)] )
val = [value stringValue];
else if( [value isKindOfClass:[NSString class]] )
val = value;
else
[NSException raise:@"Invalid format for value" format:@"Invalid format for value. addValue"];
[self addValue:val key:key];
const char * data = [val UTF8String];
CC_MD5_Update( &md5Ctx, data, strlen(data) );
}
-(void) addValue:(NSString*)value key:(NSString*) key
{
NSString *encodedValue = [self encodeData:value];
NSString *encodedKey = [self encodeData:key];
[bodyValues addObject: [NSString stringWithFormat:@"%@=%@", encodedKey, encodedValue] ];
}
-(NSData*) getBodyValues {
NSMutableData *data = [[NSMutableData alloc] init];
BOOL first=YES;
for( NSString *s in bodyValues ) {
if( !first)
[data appendBytes:"&" length:1];
[data appendBytes:[s UTF8String] length:[s length]];
first = NO;
}
return [data autorelease];
}
-(NSString*) getHashForData
{
NSString *ret;
unsigned char pTempKey[16];
// update the hash with the secret key
const char *data = [gameKey UTF8String];
CC_MD5_Update(&md5Ctx, data, strlen(data));
// then get the hash
CC_MD5_Final( pTempKey, &md5Ctx);
// NSData *nsdata = [NSData dataWithBytes:pTempKey length:16];
ret = [NSString stringWithString:@""];
for( int i=0;i<16;i++) {
ret = [NSString stringWithFormat:@"%@%02x", ret, pTempKey[i] ];
}
return ret;
}
-(NSString*) encodeData:(NSString*) data
{
NSString *newData;
newData = [data stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
// '&' and '=' should be encoded manually
newData = [newData stringByReplacingOccurrencesOfString:@"&" withString:@"%26"];
newData = [newData stringByReplacingOccurrencesOfString:@"=" withString:@"%3D"];
return newData;
}
#pragma mark NSURLConnection Delegate
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
// this method is called when the server has determined that it
// has enough information to create the NSURLResponse
// it can be called multiple times, for example in the case of a
// redirect, so each time we reset the data.
// receivedData is declared as a method instance elsewhere
[receivedData setLength:0];
}
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
// append the new data to the receivedData
// receivedData is declared as a method instance elsewhere
[receivedData appendData:data];
// NSString *dataString = [NSString stringWithCString:[data bytes] length: [data length]];
// CCLOG( @"data: %@", dataString);
}
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
CCLOG(@"Connection failed");
// wifi problems ?
postStatus_ = kPostStatusConnectionFailed;
// release the connection
self.connection = nil;
if( [delegate respondsToSelector:@selector(scorePostFail:) ] )
[delegate scorePostFail:self];
}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
// release the connection
self.connection = nil;
// NSString *dataString = [NSString stringWithCString:[receivedData bytes] length: [receivedData length]];
NSString *dataString = [NSString stringWithCString:[receivedData bytes] encoding: NSASCIIStringEncoding];
if( [dataString hasPrefix:@"OK:"] ) {
// parse ranking and other possible answers
NSArray *values = [dataString componentsSeparatedByString:@":"];
NSArray *answer = [ [values objectAtIndex:1] componentsSeparatedByString:@","];
NSEnumerator *nse = [answer objectEnumerator];
// Create a holder for each line we are going to work with
NSString *line;
// Loop through all the lines in the lines array processing each one
while( (line = [nse nextObject]) ) {
NSArray *keyvalue = [line componentsSeparatedByString:@"="];
// NSLog(@"%@",keyvalue);
if( [[keyvalue objectAtIndex:0] isEqual:@"ranking"] ) {
ranking_ = [[keyvalue objectAtIndex:1] intValue];
} else if( [[keyvalue objectAtIndex:0] isEqual:@"score_updated"] ) {
scoreDidUpdate_ = [[keyvalue objectAtIndex:1] boolValue];
}
}
if( [delegate respondsToSelector:@selector(scorePostOk:) ] )
[delegate scorePostOk:self];
} else if( [dataString hasPrefix: @"OK"] ) {
// Ok
postStatus_ = kPostStatusOK;
if( [delegate respondsToSelector:@selector(scorePostOk:) ] )
[delegate scorePostOk:self];
} else {
NSLog(@"cocoslive: Post Score failed. Reason: %@", dataString);
// Error parsing answer
postStatus_ = kPostStatusPostFailed;
if( [delegate respondsToSelector:@selector(scorePostFail:) ] )
[delegate scorePostFail:self];
}
}
-(NSURLRequest *)connection:(NSURLConnection *)connection
willSendRequest:(NSURLRequest *)request
redirectResponse:(NSURLResponse *)redirectResponse
{
NSURLRequest *newRequest=request;
if (redirectResponse) {
newRequest=nil;
}
return newRequest;
}
@end
|