I have an iPhone game in Appstore, I recently tried to upload an updated build with latest Xcode but it was rejected because the inApp-purchases were not working on ipv6 only network. It's working fine with ipv4 network.
//
// ViewController.m
//
NSMutableArray * arrayOfSection;
NSMutableArray * sectionHeaders;
NSString *error;
#import "CoinsController.h"
#import "NSString+SBJSON.h"
#import <CommonCrypto/CommonDigest.h>
@implementation CoinsController
@synthesize bkg;
-(void) callPurchaseId:(NSString*)iapId amount:(NSUInteger)ncoins{
SKPayment *payment = [SKPayment paymentWithProductIdentifier:iapId];
//[[SKPaymentQueue defaultQueue] addTransactionObserver:self];
[[SKPaymentQueue defaultQueue] addPayment:payment];
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.9;
[UIView commitAnimations];
loadingtext.text = @"Processing Purchase";
NSLog(@"Processing Purhcase");
// currentGem = [NSString stringWithFormat:@"vegas%@", @"80k"];
[CommonUtilities encryptString:[NSString stringWithFormat:@"%lu", (unsigned long)ncoins]:@"c"];
NSLog(@"Processing");
}
- (IBAction)purchaseCoins:(id)sender{
UIButton *button = (UIButton *)sender;
NSLog(@"%li", (long)button.tag);
switch (button.tag) {
case 1001:
[self callPurchaseId:IAP1 amount:IAP_AMT_1];
break;
case 1002:
[self callPurchaseId:IAP2 amount:IAP_AMT_2];
break;
case 1003:
[self callPurchaseId:IAP3 amount:IAP_AMT_3];
break;
case 1004:
[self callPurchaseId:IAP4 amount:IAP_AMT_4];
break;
case 1005:
[self callPurchaseId:IAP5 amount:IAP_AMT_5];
break;
case 1006:
[self callPurchaseId:IAP6 amount:IAP_AMT_6];
break;
default:
break;
}
}
-(void) viewWillAppear:(BOOL)animated{
[super viewWillAppear:animated];
NSNumberFormatter* nf = [[NSNumberFormatter alloc] init];
nf.usesGroupingSeparator = YES;
nf.groupingSize = 3;
((UILabel*)[self.view viewWithTag:2001]).text = [NSString stringWithFormat:@"%@ Coins", [nf stringFromNumber:[NSNumber numberWithInteger:IAP_AMT_1]]];
((UILabel*)[self.view viewWithTag:2002]).text = [NSString stringWithFormat:@"%@ Coins", [nf stringFromNumber:[NSNumber numberWithInteger:IAP_AMT_2]]];
((UILabel*)[self.view viewWithTag:2003]).text = [NSString stringWithFormat:@"%@ Coins", [nf stringFromNumber:[NSNumber numberWithInteger:IAP_AMT_3]]];
((UILabel*)[self.view viewWithTag:2004]).text = [NSString stringWithFormat:@"%@ Coins", [nf stringFromNumber:[NSNumber numberWithInteger:IAP_AMT_4]]];
((UILabel*)[self.view viewWithTag:2005]).text = [NSString stringWithFormat:@"%@ Coins", [nf stringFromNumber:[NSNumber numberWithInteger:IAP_AMT_5]]];
((UILabel*)[self.view viewWithTag:2006]).text = [NSString stringWithFormat:@"%@ Coins", [nf stringFromNumber:[NSNumber numberWithInteger:IAP_AMT_6]]];
[nf release];
}
- (void)requestProUpgradeProductData
{
NSLog(@"called productsRequest");
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.9;
[UIView commitAnimations];
loadingtext.text = @"Connecting to Store";
if([cost1000.text isEqual:@"0.00"]){
NSSet *productIdentifiers = [NSSet setWithObjects:IAP1, IAP2, IAP3, IAP4, IAP5, IAP6, nil];
SKProductsRequest *productsRequest = [[SKProductsRequest alloc] initWithProductIdentifiers:productIdentifiers];
productsRequest.delegate = self;
[productsRequest start];
}else{
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.0;
[UIView commitAnimations];
}
}
- (void)productsRequest:(SKProductsRequest *)request didReceiveResponse:(SKProductsResponse *)response
{
NSString *sym = @"";
NSArray *items = response.products;
for(SKProduct *itemproduct in items)
{
NSLog(@"Product title: %@ - %@" , itemproduct.localizedTitle, itemproduct.priceAsString);
//NSLog(@"Product description: %@" , item.localizedDescription);
//NSLog(@"Product price: " , ;
NSLog(@"Product id: %@" , itemproduct.productIdentifier);
if([itemproduct.productIdentifier isEqual:IAP1]){
cost1000.text = [sym stringByAppendingString:[NSString stringWithFormat:@"%@", itemproduct.priceAsString]];
}else if([itemproduct.productIdentifier isEqual:IAP2]){
cost3200.text = [sym stringByAppendingString:[NSString stringWithFormat:@"%@", itemproduct.priceAsString]];
}else if([itemproduct.productIdentifier isEqual:IAP3]){
cost8000.text = [sym stringByAppendingString:[NSString stringWithFormat:@"%@", itemproduct.priceAsString]];
}else if([itemproduct.productIdentifier isEqual:IAP4]){
cost20000.text = [sym stringByAppendingString:[NSString stringWithFormat:@"%@", itemproduct.priceAsString]];
}else if([itemproduct.productIdentifier isEqual:IAP5]){
cost80000.text = [sym stringByAppendingString:[NSString stringWithFormat:@"%@", itemproduct.priceAsString]];
}else if([itemproduct.productIdentifier isEqual:IAP6]){
cost200000.text = [sym stringByAppendingString:[NSString stringWithFormat:@"%@", itemproduct.priceAsString]];
}
}
for (NSString *invalidProductId in response.invalidProductIdentifiers)
{
NSLog(@"Invalid product id: %@" , invalidProductId);
error = @"YES";
}
if([error isEqual:@"YES"]){
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Payments Error Occured" message:@"Could not read payment information from Apple In-App Servers. Please try again later" delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles:nil, nil];
[alert show];
[alert release];
[self dismissViewControllerAnimated:YES completion:nil];
error = @"NO";
}
// finally release the reqest we alloc/init’ed in requestProUpgradeProductData
//[productsRequest release];
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.0;
[UIView commitAnimations];
}
//
// removes the transaction from the queue and posts a notification with the transaction result
//
- (void)finishTransaction:(SKPaymentTransaction *)transaction wasSuccessful:(BOOL)wasSuccessful
{
// remove the transaction from the payment queue.
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
if (wasSuccessful)
{
#ifdef kURL_VERIFY_PURCHASE_RECEIPT
NSString *jsonObjectString = [CommonUtilities encode:(uint8_t *)transaction.transactionReceipt.bytes length:transaction.transactionReceipt.length];
////NSLog(jsonObjectString);
[CommonUtilities encryptString:[CommonUtilities md5:jsonObjectString]:@"b"];
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.9;
[UIView commitAnimations];
//loadingtext.text = @"Purchase Completing";
loadingtext.text = @"Completing Transaction";
NSString *httpBodyString=[[NSString alloc] initWithFormat:@"receipt=%@&userid=%@",jsonObjectString, [CommonUtilities decryptString:@"username"]];
NSString *urlString=kURL_VERIFY_PURCHASE_RECEIPT;
NSURL *url=[[NSURL alloc] initWithString:urlString];
[urlString release];
NSMutableURLRequest *urlRequest=[NSMutableURLRequest requestWithURL:url];
[url release];
NSString *postLength = [NSString stringWithFormat:@"%d", [httpBodyString length]];
[urlRequest setValue:postLength forHTTPHeaderField:@"Content-Length"];
[urlRequest setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"];
[urlRequest setHTTPMethod:@"POST"];
[urlRequest setHTTPBody:[httpBodyString dataUsingEncoding:NSISOLatin1StringEncoding]];
[httpBodyString release];
[UIApplication sharedApplication].networkActivityIndicatorVisible = YES;
connection = [[NSURLConnection alloc] initWithRequest:urlRequest delegate:self];
responseData=[[NSMutableData data] retain];
#else
int addCoins = [[CommonUtilities decryptString:@"c"] intValue];
int currentCoins = [[CommonUtilities decryptString:@"coins"] intValue];
int newCoins = addCoins + currentCoins;
[CommonUtilities encryptString:[NSString stringWithFormat:@"%i", newCoins]:@"coins"];
// due to sync issues we add + 1 to xp so sync to server completes
NSLog(@"transaction complete");
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.0;
[UIView commitAnimations];
[UIApplication sharedApplication].networkActivityIndicatorVisible = NO;
#endif
}
else
{
NSLog(@"Purchase failed");
// send out a notification for the failed transaction
NSDictionary *userInfo = [NSDictionary dictionaryWithObjectsAndKeys:transaction, @"transaction" , nil];
//[[NSNotificationCenter defaultCenter] postNotificationName:kInAppPurchaseManagerTransactionFailedNotification object:self userInfo:userInfo];
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.0;
[UIView commitAnimations];
}
}
//
// called when the transaction was successful
//
- (void)completeTransaction:(SKPaymentTransaction *)transaction
{
[self finishTransaction:transaction wasSuccessful:YES];
}
//
// called when a transaction has failed
//
- (void)failedTransaction:(SKPaymentTransaction *)transaction
{
if (transaction.error.code != SKErrorPaymentCancelled)
{
// error!
[self finishTransaction:transaction wasSuccessful:NO];
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.0;
[UIView commitAnimations];
NSLog(@"error transaction");
}
else
{
// this is fine, the user just cancelled, so don’t notify
[[SKPaymentQueue defaultQueue] finishTransaction:transaction];
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.0;
[UIView commitAnimations];
NSLog(@"cancelled transaction");
}
}
//
// called when the transaction status is updated
//
- (void)paymentQueue:(SKPaymentQueue *)queue updatedTransactions:(NSArray *)transactions
{
for (SKPaymentTransaction *transaction in transactions)
{
switch (transaction.transactionState)
{
case SKPaymentTransactionStatePurchased:
[self completeTransaction:transaction];
break;
case SKPaymentTransactionStateFailed:
[self failedTransaction:transaction];
break;
default:
break;
}
}
}
#pragma mark -
- (IBAction)closeCoins:(id)sender{
[self dismissViewControllerAnimated:YES completion:nil];
}
- (void)viewDidLoad
{
[[SKPaymentQueue defaultQueue] addTransactionObserver:self];
webView.opaque = NO;
webView.backgroundColor = [UIColor clearColor];
webView.dataDetectorTypes = UIDataDetectorTypeLink;
jackpotView.opaque = NO;
jackpotView.backgroundColor = [UIColor clearColor];
//[self didLoad];
NSString *filename = @"store.jpg";
CGRect screenRect = [[UIScreen mainScreen] bounds];
if (screenRect.size.width == 568.0f) {
filename = [filename stringByReplacingOccurrencesOfString:@".jpg" withString:@"[email protected]"];
[self.bkg setBounds:CGRectMake(0, 0, screenRect.size.width, screenRect.size.height)];
NSLog(@"YAGO - changing background: %@", filename);
}
if ( UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad ) {
filename = @"store-ipad.jpg";
}
self.bkg.image = [UIImage imageNamed:filename];
viewLoading.alpha = 0.0;
viewNoInternet.alpha = 0.0;
// reachability
[self tryConnect:nil];
if([SKPaymentQueue canMakePayments]){
////NSLog(@"can make payments");
[self requestProUpgradeProductData];
}else{
////NSLog(@"cannot make payments");
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Payments Disabled" message:@"In-App Purchases are disabled on this device." delegate:nil cancelButtonTitle:@"Ok" otherButtonTitles:nil, nil];
[alert show];
[alert release];
[self dismissViewControllerAnimated: YES completion:nil];
}
CGSize result = [[UIScreen mainScreen] bounds].size;
CGFloat scale = [UIScreen mainScreen].scale;
result = CGSizeMake(result.width * scale, result.height * scale);
if(result.height == 960){
bar.frame = CGRectMake(0,0,(result.height / 2),32);
}else if(result.height == 1136){
bar.frame = CGRectMake(0,0,(result.height / 2),32);
}else if(result.height == 1024){
bar.frame = CGRectMake(0,0,result.height,44);
}else if (result.height == 2048){
bar.frame = CGRectMake(0,0,result.height/2,44);
}
else{
bar.frame = CGRectMake(0,0,(result.height),32);
}
int fontSize = 16;
if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad){
//[[UILabel appearance] setFont:[UIFont fontWithName:@"Myriad Web Pro" size:28.0]];
}else{
//[[UILabel appearance] setFont:[UIFont fontWithName:@"Myriad Web Pro" size:12.0]];
fontSize = 12;
}
UIBarButtonItem *backButton = [[[UIBarButtonItem alloc] initWithTitle:@"Close" style:UIBarButtonItemStyleBordered target:nil action:nil] autorelease];
UIImage *buttonBack32 = [[UIImage imageNamed:@"NavigationBackButton"]
resizableImageWithCapInsets:UIEdgeInsetsMake(0, 10, 0, 5)];
[[UIBarButtonItem appearance] setBackButtonBackgroundImage:buttonBack32 forState:UIControlStateNormal barMetrics:UIBarMetricsDefault];
[[UIBarButtonItem appearance] setTitleTextAttributes:
[NSDictionary dictionaryWithObjectsAndKeys:
[UIColor whiteColor],
UITextAttributeTextColor,
[UIColor colorWithRed:0.0 green:0.0 blue:0.0 alpha:0.8],
UITextAttributeTextShadowColor,
[NSValue valueWithUIOffset:UIOffsetMake(0, -1)],
UITextAttributeTextShadowOffset,
[UIFont fontWithName:@"Helvetica-Bold" size:fontSize],
UITextAttributeFont,
nil]
forState:UIControlStateNormal];
self.navigationItem.backBarButtonItem = backButton;
}
-(void)viewWillDisappear:(BOOL)animated{
NSLog(@"coins has gone");
[[NSNotificationCenter defaultCenter] postNotificationName:@"updateCoins" object:self];
[[NSNotificationCenter defaultCenter] postNotificationName:@"sortCoins" object:self];
if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad){
//[[UILabel appearance] setFont:[UIFont fontWithName:@"Myriad Web Pro" size:20.0]];
}else{
//[[UILabel appearance] setFont:[UIFont fontWithName:@"Myriad Web Pro" size:10.0]];
}
}
- (NSUInteger)supportedInterfaceOrientations {
return UIInterfaceOrientationMaskLandscape;
}
- (BOOL) shouldAutorotate {
return YES;
}
- (BOOL)shouldAutorotateToInterfaceOrientation: (UIInterfaceOrientation)interfaceOrientation {
// Return YES for supported orientations
if(interfaceOrientation == UIInterfaceOrientationLandscapeLeft || interfaceOrientation == UIInterfaceOrientationLandscapeRight){
return YES;
}else{
return NO;
}
}
- (void)dealloc {
[[SKPaymentQueue defaultQueue] removeTransactionObserver:self];
[self.bkg release];
[super dealloc];
}
#pragma mark for server side validation (no customer support)
// these connection-related methods are only here to help in case you ever decide to implement
// server-side validation of purchase receipts
// - in this case, you'll also need to define your own kURL_VERIFY_PURCHASE_RECEIPT and figure out the existing validation protocol (I can't provide you with support on this one, it's code written by someone else)
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
[responseData setLength:0];
//NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse*)response;
if ([response respondsToSelector:@selector(allHeaderFields)]) {
//NSDictionary *dictionary = [httpResponse allHeaderFields];
//NSLog([dictionary description]);
}
}
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {
[responseData appendData:data];
//NSString *a = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
//NSLog(@"Data: %@", a);
}
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"No Connection" message:@"You are not connected to the internet or data. Please connect and try again." delegate:nil cancelButtonTitle:@"Cancel" otherButtonTitles:nil, nil];
[alert show];
[alert release];
}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
NSString *responseString = [[NSString alloc] initWithData:responseData encoding:NSUTF8StringEncoding];
//////NSLog(responseString);
NSString *gKey = [CommonUtilities decryptString:@"b"];
NSString *result = [CommonUtilities base64Decrypt:responseString:gKey];
NSLog(@"%@",result);
int r = [result intValue];
////NSLog(@"r int: %i", r);
if(r == 1){
int addCoins = [[CommonUtilities decryptString:@"c"] intValue];
int currentCoins = [[CommonUtilities decryptString:@"coins"] intValue];
int newCoins = addCoins + currentCoins;
[CommonUtilities encryptString:[NSString stringWithFormat:@"%i", newCoins]:@"coins"];
// due to sync issues we add + 1 to xp so sync to server completes
NSLog(@"transaction complete");
}else{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Receipt Failed" message:@"Please make the purchase again using a vaild iTunes account" delegate:nil cancelButtonTitle:@"Cancel" otherButtonTitles:nil, nil];
[alert show];
[alert release];
}
[self dismissViewControllerAnimated:YES completion:nil];
[UIView setAnimationDuration:0.5];
[UIView setAnimationDelegate:self];
[UIView setAnimationDidStopSelector:@selector(fadeOut:finished:context:)];
viewLoading.alpha = 0.0;
[UIView commitAnimations];
[UIApplication sharedApplication].networkActivityIndicatorVisible = NO;
}
@end
I don't know whats wrong with this code. The only thing I get in the xcode debug log is "Purchase failed" . Any help will be highly appreciated.
Turns out the problem wasn't on my side . Since sandbox.itunes.apple.com does not have an ipv6 address so it works with DNS64 (or NAT64 or whatever this thing is :D .I don't know much about this stuff) and not on a real ipv6 only network.
Not sure why they rejected my app the first time but it was approved after I resubmitted it .
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With