Last active
August 29, 2015 13:57
-
-
Save itsbalamurali/9909948 to your computer and use it in GitHub Desktop.
Replace index.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?php | |
$_SERVER['SERVER_PORT'] = 80; | |
/** | |
* Step 1: Require the Slim Framework | |
* | |
* If you are not using Composer, you need to require the | |
* Slim Framework and register its PSR-0 autoloader. | |
* | |
* If you are using Composer, you can skip this step. | |
*/ | |
require 'Slim/Slim.php'; | |
// Pimple Dependency Injection Container | |
require_once 'Pimple.php'; | |
\Slim\Slim::registerAutoloader(); | |
$app = new \Slim\Slim(); | |
// Set Default Timezone to America/Los_Angeles as this is what all App Store dates are in. | |
date_default_timezone_set('America/Los_Angeles'); | |
// Create Pimple Dependency Injection Container for DB Connection. | |
$dbContainer = new Pimple(); | |
// Global Development Mode Setting | |
$developmentMode = ''; | |
// Global iTunes Production Level Setting | |
$iTunesProductionLevel = ''; | |
// Global iTunes Receipt Validation Caching Setting | |
$iTunesCachingDuration = -1; | |
// Global iTunes Receipt Validation Caching Setting | |
$subscriptionBehavior = ''; | |
// API Version | |
$apiVersion = '1.1'; | |
// DB Setup for Pimple Container | |
// BakerCloud API SETUP CONFIGURATION SETTING | |
// ************************************************************ | |
$dbContainer['db.options'] = array( | |
'host' => getenv('CLOUD_SQL_INSTANCE'), // CONFIGURE TO YOUR DB HOSTNAME | |
'username' => getenv('DB_USERNAME'), // CONFIGURE TO YOUR DB USERNAME | |
'password' => '', // CONFIGURE TO YOUR DB USERNAME'S PASSWORD | |
'dbname' => getenv('DB_NAME') // CONFIGURE TO YOUR DB INSTANCE NAME | |
); | |
//************************************************************* | |
// Using "share" method makes sure that the function is only called when 'db' is retrieved the first time. | |
$dbContainer['db'] = $dbContainer->share(function () use($dbContainer) | |
{ | |
// Get DB handle and create new PDO DB object using configuration settings | |
//$dbHandle = new PDO('mysql:host=' . $dbContainer['db.options']['host'] . ';dbname=' . $dbContainer['db.options']['dbname'], $dbContainer['db.options']['username'], $dbContainer['db.options']['password']); | |
$dbHandle = new PDO('mysql:unix_socket=' . $dbContainer['db.options']['host'] . ';dbname=' . $dbContainer['db.options']['dbname'], $dbContainer['db.options']['username'], $dbContainer['db.options']['password']); | |
// Set Character Set for DB connection to UTF-8 | |
$dbHandle -> exec("SET CHARACTER SET utf8"); | |
// Return DB handle | |
return $dbHandle; | |
}); | |
// ************************************************ | |
// SLIM PHP Methods for handling REST API Methods | |
// ************************************************ | |
// GET route | |
$app->get( | |
'/', | |
function () { | |
$template = <<<EOT | |
<!DOCTYPE html> | |
<html> | |
<head> | |
<meta charset='utf-8'/> | |
<title>Baker Cloud Console (CE) REST API</title> | |
<style> | |
html,body,div,span,object,iframe, | |
h1,h2,h3,h4,h5,h6,p,blockquote,pre, | |
abbr,address,cite,code, | |
del,dfn,em,img,ins,kbd,q,samp, | |
small,strong,sub,sup,var, | |
b,i, | |
dl,dt,dd,ol,ul,li, | |
fieldset,form,label,legend, | |
table,caption,tbody,tfoot,thead,tr,th,td, | |
article,aside,canvas,details,figcaption,figure, | |
footer,header,hgroup,menu,nav,section,summary, | |
time,mark,audio,video{margin:0;padding:0;border:0;outline:0;font-size:100%;vertical-align:baseline;background:transparent;} | |
body{line-height:1;} | |
article,aside,details,figcaption,figure, | |
footer,header,hgroup,menu,nav,section{display:block;} | |
nav ul{list-style:none;} | |
blockquote,q{quotes:none;} | |
blockquote:before,blockquote:after, | |
q:before,q:after{content:'';content:none;} | |
a{margin:0;padding:0;font-size:100%;vertical-align:baseline;background:transparent;} | |
ins{background-color:#ff9;color:#000;text-decoration:none;} | |
mark{background-color:#ff9;color:#000;font-style:italic;font-weight:bold;} | |
del{text-decoration:line-through;} | |
abbr[title],dfn[title]{border-bottom:1px dotted;cursor:help;} | |
table{border-collapse:collapse;border-spacing:0;} | |
hr{display:block;height:1px;border:0;border-top:1px solid #cccccc;margin:1em 0;padding:0;} | |
input,select{vertical-align:middle;} | |
html{ background: #EDEDED; height: 100%; } | |
body{background:#FFF;margin:0 auto;min-height:100%;padding:0 30px;width:440px;color:#666;font:14px/23px Arial,Verdana,sans-serif;} | |
h1,h2,h3,p,ul,ol,form,section{margin:0 0 20px 0;} | |
h1{color:#333;font-size:20px;} | |
h2,h3{color:#333;font-size:14px;} | |
h3{margin:0;font-size:12px;font-weight:bold;} | |
ul,ol{list-style-position:inside;color:#999;} | |
ul{list-style-type:square;} | |
code,kbd{background:#EEE;border:1px solid #DDD;border:1px solid #DDD;border-radius:4px;-moz-border-radius:4px;-webkit-border-radius:4px;padding:0 4px;color:#666;font-size:12px;} | |
pre{background:#EEE;border:1px solid #DDD;border-radius:4px;-moz-border-radius:4px;-webkit-border-radius:4px;padding:5px 10px;color:#666;font-size:12px;} | |
pre code{background:transparent;border:none;padding:0;} | |
a{color:#0068ae;} | |
header{padding: 30px 0;text-align:center;} | |
</style> | |
</head> | |
<body> | |
<header> | |
<a href='http://www.bakerframework.com'><img src='' alt='Baker Cloud (CE)'/></a> | |
</header> | |
<h1>Welcome to Baker Cloud Console (CE) API!</h1> | |
<p> | |
Congratulations! Your Baker Cloud Console (CE) API is up and running. However, you should test the Database connectivity below. If you get a success message, then you should be good to go! | |
</p> | |
<section> | |
<h2>Get Started</h2> | |
<ol> | |
<li><a href='/checkinstall' target='_blank'>Check DB Connectivity</a></li> | |
<li>Read the <a href='http://www.bakerframework.com/' target='_blank'>online documentation</a></li> | |
<li>Follow <a href='http://www.twitter.com/bakerframework' target='_blank'>@bakerframework</a> on Twitter</li> | |
</ol> | |
</section> | |
<section> | |
<h2>Baker Framework Community</h2> | |
<h3>Support Forum and Knowledge Base</h3> | |
<p> | |
Visit the <a href='http://www.github.com/bakerframework/' target='_blank'>Online Support Community</a> | |
to read announcements, chat with fellow Baker Framework users, ask questions, help others, or show off your cool | |
Baker Newsstand apps. | |
</p> | |
<h3>Twitter</h3> | |
<p> | |
Follow <a href='http://www.twitter.com/bakerframework' target='_blank'>@bakerframework</a> on Twitter to receive the very latest news | |
and updates about the framework. | |
</p> | |
</section> | |
</body> | |
</html> | |
EOT; | |
echo $template; | |
} | |
); | |
// Check DB Connectivity | |
// *Makes connection to BakerCloud DB and tries to make a select from the PUBLICATION table | |
$app->get('/checkinstall/', function () | |
{ | |
try { | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
$result = $db->prepare("SELECT * FROM PUBLICATION"); | |
$result->execute(); | |
$checkInstall = $result->fetchAll(); | |
echo '{"BakerCloud API":{"Success":"Database Connection Test Successful"}}'; | |
} | |
catch(PDOException $e) { | |
echo '{"BakerCloud API":{"Error":"' . $e->getMessage() . '"}}'; | |
} | |
}); | |
// Output Debug Information | |
// *Outputs the settings for a given publication that the API is registering, useful when debugging | |
$app->get('/debuginformation/:app_id', function ($app_id) | |
{ | |
global $apiVersion; | |
try { | |
echo 'Baker Cloud CE API: Debug Information'; | |
echo '<BR>API Version: ' . $apiVersion; | |
echo '<BR>Development Mode: ' . isInDevelopmentMode($app_id); | |
echo '<BR>Subscription Behavior: ' . getSubscriptionBehavior($app_id); | |
echo '<BR>iTunes Production Level: ' . getiTunesProductionLevel($app_id); | |
echo '<BR>iTunes Caching Duration: ' . getiTunesCachingDuration($app_id); | |
} | |
catch(PDOException $e) { | |
} | |
}); | |
// Issues List | |
// *Retrieves a list of available issues for the App ID, for population of Baker Shelf | |
$app->get('/issues/:app_id/:user_id', function ($app_id, $user_id) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
$SCRIPT_LOCATION = str_replace('index.php','',$_SERVER['SCRIPT_NAME']); | |
// Lookup Issue Download Security condition for Publication, if true, create secured API Issue download links | |
$result = $db->query("SELECT ISSUE_DOWNLOAD_SECURITY FROM PUBLICATION WHERE APP_ID = '$app_id' LIMIT 0, 1"); | |
$issueDownloadSecurity = $result->fetchColumn(); | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Retrieving Issues for APP ID: " . $app_id . " USER ID: " . $user_id);} | |
// Query all issues for the incoming APP_ID | |
$sql = "SELECT * FROM ISSUES WHERE APP_ID = '$app_id' AND AVAILABILITY = 'published'"; | |
try { | |
$IssuesArray = array(); | |
$i = 0; | |
foreach($db->query($sql) as $row) { | |
$IssuesArray[$i]['name'] = $row['NAME']; | |
$IssuesArray[$i]['title'] = $row['TITLE']; | |
$IssuesArray[$i]['info'] = $row['INFO']; | |
$IssuesArray[$i]['date'] = $row['DATE']; | |
$IssuesArray[$i]['cover'] = $row['COVER']; | |
if ($issueDownloadSecurity == "TRUE") { | |
$IssuesArray[$i]['url'] = "http://" . $_SERVER['HTTP_HOST'] . $SCRIPT_LOCATION . "issue/" . $app_id . "/" . $user_id . "/" . $row['NAME']; | |
} | |
else{ | |
$IssuesArray[$i]['url'] = $row['URL']; | |
} | |
if($row['PRICING'] != 'free') | |
{ | |
$IssuesArray[$i]['product_id'] = $row['PRODUCT_ID']; | |
} | |
$i++; | |
} | |
logAnalyticMetric(AnalyticType::ApiInteraction,1,NULL,$app_id,$user_id); | |
echo json_encode($IssuesArray); | |
} | |
catch(PDOException $e) { | |
logMessage(LogType::Error, $e->getMessage()); | |
echo '{"error":{"text":"' . $e->getMessage() . '"}}'; | |
} | |
}); | |
// Issue Download | |
// *Validates availability of download of a specific named issue, redirects to download if available | |
$app->get('/issue/:app_id/:user_id/:name', function ($app_id, $user_id, $name) use($app) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
try { | |
$result = $db->prepare("SELECT * FROM ISSUES WHERE APP_ID = '$app_id' AND NAME = '$name' LIMIT 0,1"); | |
$result->execute(); | |
$issue = $result->fetch(); | |
// The Issue is not found for App_ID and Name. Throw 404 not found error. | |
if (!$issue) { | |
header('HTTP/1.1 404 Not Found'); | |
die(); | |
} | |
// Retrieve issue Issue Product ID to cross check with purchases | |
$product_id = $issue['PRODUCT_ID']; | |
// Default to not allow download. | |
$allow_download = false; | |
// Validate that the Product ID (from Issue Name) is an available download for given user | |
if ($product_id && $issue['PRICING'] != 'free') { | |
// Allow download if the issue is marked as purchased | |
$result = $db->query("SELECT COUNT(*) FROM PURCHASES | |
WHERE APP_ID = '$app_id' AND USER_ID = '$user_id' AND PRODUCT_ID = '$product_id'"); | |
$allow_download = ($result->fetchColumn() > 0); | |
} else if ($issue['PRICING'] == 'free') { | |
// Issue is marked as free, allow download | |
$allow_download = true; | |
} | |
if ($allow_download) { | |
if((isInDevelopmentMode($app_id)=="TRUE") && !($app->request()->isHead())){logMessage(LogType::Info,"Downloading ISSUE: " . $name . " for APP ID: " . $app_id . " USER ID: " . $user_id);} | |
logAnalyticMetric(AnalyticType::ApiInteraction,1,NULL,$app_id,$user_id); | |
if(!($app->request()->isHead())){logAnalyticMetric(AnalyticType::Download,1,$name,$app_id,$user_id);} | |
// Redirect to the downloadable file, nothing else needed in API call | |
$app->response()->redirect($issue['URL'], 303); | |
} | |
else { | |
header('HTTP/1.1 403 Forbidden'); | |
die(); | |
} | |
} | |
catch(PDOException $e) { | |
// Handle exception | |
logMessage(LogType::Error, $e->getMessage()); | |
} | |
}); | |
// Purchases List | |
// *Returns a list of Purchased Product ID's | |
$app->get('/purchases/:app_id/:user_id', function ($app_id, $user_id) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
$purchased_product_ids = array(); | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Checking purchases for APP ID: " . $app_id . " USER ID: " . $user_id);} | |
try { | |
$subscribed = false; | |
// Retrieve latest receipt for Auto-Renewable-Subscriptions for the APP_ID, USER_ID combination | |
$result = $db->query("SELECT BASE64_RECEIPT FROM RECEIPTS | |
WHERE APP_ID = '$app_id' AND USER_ID = '$user_id' AND TYPE = 'auto-renewable-subscription' | |
ORDER BY TRANSACTION_ID DESC LIMIT 0, 1"); | |
$base64_latest_receipt = $result->fetchColumn(); | |
if($base64_latest_receipt) | |
{ | |
$userSubscription = checkSubscription($app_id, $user_id); | |
$dateLastValidated = new DateTime($userSubscription["LAST_VALIDATED"]); | |
$dateExpiration = new DateTime($userSubscription["EXPIRATION_DATE"]); | |
$dateCurrent = new DateTime('now'); | |
$interval = $dateCurrent->diff($dateLastValidated); | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Time since last validating receipt for APP ID: " . $app_id . " USER ID: " . $user_id . " = " . $interval->format('%h hours %i minutes') );} | |
// Only refresh and re-verify receipt if greater than the iTunesCachingDuration - or greater than 1 whole day | |
if ((getiTunesCachingDuration($app_id) == -1) || ($interval->format('%h') > getiTunesCachingDuration($app_id)) || ($interval->format('%a') > 1)) { | |
// Check the latest receipt from the subscription table | |
if ($base64_latest_receipt) { | |
// Verify Receipt - with logic to fall back to Sandbox test if Production Receipt fails (error code 21007) | |
try{ | |
$data = verifyReceipt($base64_latest_receipt, $app_id, $user_id); | |
} | |
catch(Exception $e) { | |
if($e->getCode() == "21007"){ | |
logMessage(LogType::Info,"Confirming purchase for APP ID - Sandbox Receipt used in Production, retrying against Sandbox iTunes API: " . $app_id . " USER ID: " . $user_id . " TYPE: " . $type); | |
$data = verifyReceipt($base64_latest_receipt, $app_id, $user_id, TRUE); | |
} | |
} | |
markIssuesAsPurchased($data, $app_id, $user_id); | |
// Check if there is an active subscription for the user. Status=0 is true. | |
$subscribed = ($data->status == 0); | |
} | |
else { | |
// There is no receipt for this user, there is no active subscription | |
$subscribed = false; | |
} | |
} | |
else { | |
// We aren't going to re-verify the receipt now, but we should determine if the Expiration Date is beyond now | |
if ($dateCurrent > $dateExpiration) { | |
$subscribed = false; | |
} | |
else { | |
$subscribed = true; | |
} | |
} | |
} | |
else | |
{ | |
// There is no Auto-Renewable-Subscription for the APP_ID, USER_ID combination - check if there is a Free-Subscription | |
$result = $db->query("SELECT BASE64_RECEIPT FROM RECEIPTS | |
WHERE APP_ID = '$app_id' AND USER_ID = '$user_id' AND TYPE = 'free-subscription' | |
ORDER BY TRANSACTION_ID DESC LIMIT 0, 1"); | |
$base64_latest_receipt = $result->fetchColumn(); | |
// If there is a receipt for a free-subscription then we will return that the USER_ID is subscribed. Since a free | |
// subscription really doesn't have and valid term then we will just ignore any dates in the Apple receipt. | |
// However the list of purchased product_ids can still be blank because the issues should be marked as free. | |
if($base64_latest_receipt){ | |
$subscribed = true; | |
} | |
} | |
// Return list of purchased product_ids for the user | |
$result = $db->query("SELECT PRODUCT_ID FROM PURCHASES | |
WHERE APP_ID = '$app_id' AND USER_ID = '$user_id'"); | |
$purchased_product_ids = $result->fetchAll(PDO::FETCH_COLUMN); | |
logAnalyticMetric(AnalyticType::ApiInteraction,1,NULL,$app_id,$user_id); | |
echo json_encode(array( | |
'issues' => $purchased_product_ids, | |
'subscribed' => $subscribed | |
)); | |
} | |
catch(PDOException $e) { | |
logMessage(LogType::Error, $e->getMessage()); | |
echo '{"error":{"text":"' . $e->getMessage() . '"}}'; | |
} | |
}); | |
// iTunes List | |
// *Returns a list of Issues in an iTunes ATOM Feed XML Format. This can be hooked up to the FEED URL within | |
// iTunes connect to display up to date information in the Newsstand App Store listing | |
$app->get('/itunes/:app_id', function ($app_id) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
try { | |
$result = $db->query("SELECT ITUNES_UPDATED FROM PUBLICATION WHERE APP_ID = '$app_id' LIMIT 0, 1"); | |
$ITUNES_UPDATED = $result->fetchColumn(); | |
if(!$ITUNES_UPDATED) | |
throw new Exception('Invalid APP ID'); | |
$iTunesUpdateDate = new DateTime($ITUNES_UPDATED); | |
$sql = "SELECT * FROM ISSUES WHERE APP_ID = '$app_id' AND AVAILABILITY = 'published'"; | |
$AtomXML = "<?xml version=\"1.0\" encoding=\"UTF-8\"" . "?>"; | |
$AtomXML.= "<feed xmlns=\"http://www.w3.org/2005/Atom\" xmlns:news=\"http://itunes.apple.com/2011/Newsstand\">"; | |
$AtomXML.= "<updated>" . date_format($iTunesUpdateDate, DateTime::ATOM) . "</updated>"; | |
foreach($db->query($sql) as $row) { | |
$iTunesIssueUpdateDate = new DateTime($row['ITUNES_UPDATED']); | |
$iTunesPublishedDate = new DateTime($row['DATE']); | |
$AtomXML.= "<entry>"; | |
$AtomXML.= "<id>" . $row['NAME'] . "</id>"; | |
$AtomXML.= "<updated>" . date_format($iTunesIssueUpdateDate, DateTime::ATOM) . "</updated>"; | |
$AtomXML.= "<published>" . date_format($iTunesPublishedDate, DateTime::ATOM) . "</published>"; | |
$AtomXML.= "<summary>" . $row['ITUNES_SUMMARY'] . "</summary>"; | |
$AtomXML.= "<news:cover_art_icons>"; | |
$AtomXML.= "<news:cover_art_icon size=\"SOURCE\" src=\"" . $row['ITUNES_COVERART_URL'] . "\"/>"; | |
$AtomXML.= "</news:cover_art_icons>"; | |
$AtomXML.= "</entry>"; | |
} | |
$AtomXML.= "</feed>"; | |
logAnalyticMetric(AnalyticType::ApiInteraction,1,NULL,$app_id,$user_id); | |
echo utf8_encode($AtomXML); | |
} | |
catch(Exception $e) { | |
logMessage(LogType::Error, $e->getMessage()); | |
echo '{"error":{"text":"' . $e->getMessage() . '"}}'; | |
} | |
}); | |
// Confirm Purchase | |
// *Confirms the purchase by validating the Receipt_Data received for the in app purchase. Records the receipt data | |
// in the database and adds the available issues to the user's Purchased List | |
$app->post('/confirmpurchase/:app_id/:user_id', function ($app_id, $user_id) use($app) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
$body = $app->request()->getBody(); | |
$receiptdata = $app->request()->post('receipt_data'); | |
$type = $app->request()->post('type'); | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Confirming purchase for APP ID: " . $app_id . " USER ID: " . $user_id . " TYPE: " . $type);} | |
try { | |
// Verify Receipt - with logic to fall back to Sandbox test if Production Receipt fails (error code 21007) | |
try{ | |
$iTunesReceiptInfo = verifyReceipt($receiptdata, $app_id, $user_id); | |
} | |
catch(Exception $e) { | |
if($e->getCode() == "21007"){ | |
logMessage(LogType::Info,"Confirming purchase for APP ID - Sandbox Receipt used in Production, retrying against Sandbox iTunes API: " . $app_id . " USER ID: " . $user_id . " TYPE: " . $type); | |
$iTunesReceiptInfo = verifyReceipt($receiptdata, $app_id, $user_id, TRUE); | |
} | |
} | |
$sql = "INSERT IGNORE INTO RECEIPTS (APP_ID, QUANTITY, PRODUCT_ID, TYPE, TRANSACTION_ID, USER_ID, PURCHASE_DATE, | |
ORIGINAL_TRANSACTION_ID, ORIGINAL_PURCHASE_DATE, APP_ITEM_ID, VERSION_EXTERNAL_IDENTIFIER, BID, BVRS, BASE64_RECEIPT) | |
VALUES (:app_id, :quantity, :product_id, :type, :transaction_id, :user_id, :purchase_date, :original_transaction_id, | |
:original_purchase_date, :app_item_id, :version_external_identifier, :bid, :bvrs, :base64_receipt)"; | |
try { | |
$stmt = $db->prepare($sql); | |
$stmt->bindParam("app_id", $app_id); | |
$stmt->bindParam("quantity", $iTunesReceiptInfo->receipt->quantity); | |
$stmt->bindParam("product_id", $iTunesReceiptInfo->receipt->product_id); | |
$stmt->bindParam("type", $type); | |
$stmt->bindParam("transaction_id", $iTunesReceiptInfo->receipt->transaction_id); | |
$stmt->bindParam("user_id", $user_id); | |
$stmt->bindParam("purchase_date", $iTunesReceiptInfo->receipt->purchase_date); | |
$stmt->bindParam("original_transaction_id", $iTunesReceiptInfo->receipt->original_transaction_id); | |
$stmt->bindParam("original_purchase_date", $iTunesReceiptInfo->receipt->original_purchase_date); | |
$stmt->bindParam("app_item_id", $iTunesReceiptInfo->receipt->item_id); | |
$stmt->bindParam("version_external_identifier", $iTunesReceiptInfo->receipt->version_external_identifier); | |
$stmt->bindParam("bid", $iTunesReceiptInfo->receipt->bid); | |
$stmt->bindParam("bvrs", $iTunesReceiptInfo->receipt->bvrs); | |
$stmt->bindParam("base64_receipt", $receiptdata); | |
$stmt->execute(); | |
// If successful, record the user's purchase | |
if($type == 'auto-renewable-subscription'){ | |
markIssuesAsPurchased($iTunesReceiptInfo,$app_id,$user_id); | |
}else if($type == 'issue'){ | |
markIssueAsPurchased($iTunesReceiptInfo->receipt->product_id, $app_id, $user_id); | |
}else if($type == 'free-subscription'){ | |
// Nothing to do, as the server assumes free subscriptions don't need to be handled in this way | |
} | |
logAnalyticMetric(AnalyticType::ApiInteraction,1,NULL,$app_id,$user_id); | |
} | |
catch(PDOException $e) { | |
logMessage(LogType::Error, $e->getMessage()); | |
echo '{"error":{"text":"' . $e->getMessage() . '"}}'; | |
} | |
} | |
catch(Exception $e) { | |
logMessage(LogType::Error, $e->getMessage()); | |
echo '{"error":{"text":"' . $e->getMessage() . '"}}'; | |
} | |
}); | |
// APNS Token | |
// *Stores the APNS Token in the database for the given App ID and User ID | |
$app->post('/apns/:app_id/:user_id', function ($app_id, $user_id) use($app) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
$apns_token = $app->request()->post('apns_token'); | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Storing APNS Token for APP ID: " . $app_id . " USER ID: " . $user_id);} | |
$sql = "INSERT IGNORE INTO APNS_TOKENS (APP_ID, USER_ID, APNS_TOKEN) | |
VALUES (:app_id, :user_id, :apns_token)"; | |
try { | |
$stmt = $db->prepare($sql); | |
$stmt->bindParam("app_id", $app_id); | |
$stmt->bindParam("user_id", $user_id); | |
$stmt->bindParam("apns_token", $apns_token); | |
$stmt->execute(); | |
logAnalyticMetric(AnalyticType::ApiInteraction,1,NULL,$app_id,$user_id); | |
echo '{"success":{"message":"' . $apns_token . '"}}'; | |
} | |
catch(PDOException $e) { | |
logMessage(LogType::Error, $e->getMessage()); | |
echo '{"error":{"text":"' . $e->getMessage() . '"}}'; | |
} | |
}); | |
// ************************************************ | |
// Utility Functions | |
// ************************************************ | |
// Log Error Messages for tracking and debugging purposes, also displayed in the BakerCloud Console for issue debugging | |
function logMessage($logType, $logMessage) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
$sql = "INSERT INTO SYSTEM_LOG (TYPE, MESSAGE) | |
VALUES (:logtype, :logmessage)"; | |
try { | |
$stmt = $db->prepare($sql); | |
$stmt->bindParam("logtype", $logType); | |
$stmt->bindParam("logmessage", $logMessage); | |
$stmt->execute(); | |
} | |
catch(PDOException $e) { | |
// Error occurred, just ignore because if it failed in this logMessage method not much we can do, ignore | |
} | |
} | |
// Log Analytic Metrics for tracking purposes and basic display in the dashboard | |
function logAnalyticMetric($analytic_type, $analytic_value, $metadata, $app_id, $user_id) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
$sql = "INSERT INTO ANALYTICS (APP_ID, USER_ID, TYPE, VALUE, METADATA) | |
VALUES (:app_id, :user_id, :analytic_type, :analytic_value, :metadata)"; | |
try { | |
$stmt = $db->prepare($sql); | |
$stmt->bindParam("app_id", $app_id); | |
$stmt->bindParam("user_id", $user_id); | |
$stmt->bindParam("analytic_type", $analytic_type); | |
$stmt->bindParam("analytic_value", $analytic_value); | |
$stmt->bindParam("metadata", $metadata); | |
$stmt->execute(); | |
} | |
catch(PDOException $e) { | |
// Error occurred, just ignore because if it failed in this logAnalyticMetric method not much we can do, ignore | |
} | |
} | |
// Check if this publication is in development mode, useful for installation and non-production debugging | |
function isInDevelopmentMode($app_id) | |
{ | |
global $developmentMode; | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
if($developmentMode != ""){ | |
return $developmentMode; | |
} | |
else{ | |
$result = $db->query("SELECT DEVELOPMENT_MODE FROM PUBLICATION WHERE APP_ID = '$app_id' LIMIT 0, 1"); | |
return $result->fetchColumn(); | |
} | |
} | |
// Retrieve the iTunes Production Level for Apple API calls | |
function getiTunesProductionLevel($app_id) | |
{ | |
global $iTunesProductionLevel; | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
if($iTunesProductionLevel != ""){ | |
return $iTunesProductionLevel; | |
} | |
else{ | |
$result = $db->query("SELECT ITUNES_PRODUCTION_LEVEL FROM PUBLICATION WHERE APP_ID = '$app_id' LIMIT 0, 1"); | |
return $result->fetchColumn(); | |
} | |
} | |
// Retrieve the Subscription Behavior setting for marking issues as purchased | |
function getSubscriptionBehavior($app_id) | |
{ | |
global $subscriptionBehavior; | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
if($subscriptionBehavior != ""){ | |
return $subscriptionBehavior; | |
} | |
else{ | |
$result = $db->query("SELECT SUBSCRIPTION_BEHAVIOR FROM PUBLICATION WHERE APP_ID = '$app_id' LIMIT 0, 1"); | |
return $result->fetchColumn(); | |
} | |
} | |
// Retrieve the iTunes Caching Duration for re-validating Apple Receipts | |
function getiTunesCachingDuration($app_id) | |
{ | |
global $iTunesCachingDuration; | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
if($iTunesCachingDuration != -1){ | |
return $iTunesCachingDuration; | |
} | |
else{ | |
$result = $db->query("SELECT ITUNES_REVALIDATION_DURATION FROM PUBLICATION WHERE APP_ID = '$app_id' LIMIT 0, 1"); | |
return $result->fetchColumn(); | |
} | |
} | |
// Mark all available (paid) issues as purchased for a given user | |
function markIssuesAsPurchased($app_store_data, $app_id, $user_id) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Marking Issues as Purchased for APP ID: " . $app_id . " USER ID: " . $user_id);} | |
$receipt = $app_store_data->receipt; | |
$startDate = new DateTime($receipt->purchase_date_pst); | |
if ($app_store_data->status == 0) { | |
$endDate = new DateTime($app_store_data->latest_receipt_info->expires_date_formatted_pst); | |
} | |
else | |
if ($app_store_data->status == 21006) { | |
$endDate = new DateTime($app_store_data->latest_expired_receipt_info->expires_date_formatted_pst); | |
} | |
// Now update the Purchases table with all Issues that fall within the subscription start and expiration date | |
$startDateFormatted = $startDate->format('Y-m-d H:i:s'); | |
$endDateFormatted = $endDate->format('Y-m-d H:i:s'); | |
// Get First Day of the Month that the Receipt was generated for (Start) | |
$issuesStartDateFormatted = $startDate->format('Y-m-01 00:00:00'); | |
// Get Last Day of the Month that the Receipt was generated for (Expiration) | |
$issuesEndDateFormatted = $endDate->format('Y-m-t 23:59:59'); | |
// Update Subscriptions Table for user with current active subscription start and expiration date | |
updateSubscription($app_id, $user_id, $startDateFormatted, $endDateFormatted); | |
// If we are in Sandbox Mode, unlock all issues by default for testing purposes | |
if(getiTunesProductionLevel($app_id)=="sandbox"){ | |
$result = $db->query("SELECT PRODUCT_ID FROM ISSUES | |
WHERE APP_ID = '$app_id' | |
AND PRICING = 'paid'"); | |
} | |
else{ | |
// If we are in Production - determine based on Subscription Behavior setting | |
if(getSubscriptionBehavior($app_id)=="all"){ | |
$result = $db->query("SELECT PRODUCT_ID FROM ISSUES | |
WHERE APP_ID = '$app_id' | |
AND PRICING = 'paid'"); | |
}else if(getSubscriptionBehavior($app_id)=="term"){ | |
$result = $db->query("SELECT PRODUCT_ID FROM ISSUES | |
WHERE APP_ID = '$app_id' | |
AND `DATE` >= '$issuesStartDateFormatted' | |
AND `DATE` <= '$issuesEndDateFormatted' | |
AND PRICING = 'paid' | |
AND AVAILABILITY = 'published'"); | |
}else{ | |
//Default to 'term' if for some reason the above fails | |
$result = $db->query("SELECT PRODUCT_ID FROM ISSUES | |
WHERE APP_ID = '$app_id' | |
AND `DATE` >= '$issuesStartDateFormatted' | |
AND `DATE` <= '$issuesEndDateFormatted' | |
AND PRICING = 'paid' | |
AND AVAILABILITY = 'published'"); | |
} | |
} | |
$product_ids_to_mark = $result->fetchAll(PDO::FETCH_COLUMN); | |
$insert = "INSERT IGNORE INTO PURCHASES (APP_ID, USER_ID, PRODUCT_ID) | |
VALUES ('$app_id', '$user_id', :product_id)"; | |
$stmt = $db->prepare($insert); | |
foreach($product_ids_to_mark as $key => $product_id) { | |
$stmt->bindParam('product_id', $product_id); | |
$stmt->execute(); | |
} | |
} | |
// Mark all available issues as purchased for a given user | |
function markIssueAsPurchased($product_id, $app_id, $user_id) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Marking single issue as purchased for APP ID: " . $app_id . " USER ID: " . $user_id . " PRODUCT ID: " . $product_id);} | |
$sql = "INSERT IGNORE INTO PURCHASES (APP_ID, USER_ID, PRODUCT_ID) | |
VALUES (:app_id, :user_id, :product_id)"; | |
try { | |
$stmt = $db->prepare($sql); | |
$stmt->bindParam("app_id", $app_id); | |
$stmt->bindParam("user_id", $user_id); | |
$stmt->bindParam("product_id", $product_id); | |
$stmt->execute(); | |
} | |
catch(PDOException $e) { | |
logMessage(LogType::Error, $e->getMessage()); | |
echo '{"error":{"text":"' . $e->getMessage() . '"}}'; | |
} | |
} | |
// Update the Subscription Record for a specific user with Effective Date and Expiration Date | |
function updateSubscription($app_id, $user_id, $effective_date, $expiration_date) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
$currentDate = new DateTime('now'); | |
$lastValidated = $currentDate->format('Y-m-d H:i:s'); | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Updating subscription effective dates for APP ID: " . $app_id . " USER ID: " . $user_id);} | |
$sql = "INSERT INTO SUBSCRIPTIONS (APP_ID, USER_ID, EFFECTIVE_DATE, EXPIRATION_DATE, LAST_VALIDATED) | |
VALUES (:app_id, :user_id, :effective_date, :expiration_date, :last_validated) | |
ON DUPLICATE KEY UPDATE EFFECTIVE_DATE=:effective_date, EXPIRATION_DATE=:expiration_date, LAST_VALIDATED=:last_validated"; | |
try { | |
$stmt = $db->prepare($sql); | |
$stmt->bindParam("app_id", $app_id); | |
$stmt->bindParam("user_id", $user_id); | |
$stmt->bindParam("effective_date", $effective_date); | |
$stmt->bindParam("expiration_date", $expiration_date); | |
$stmt->bindParam("last_validated", $lastValidated); | |
$stmt->execute(); | |
} | |
catch(PDOException $e) { | |
logMessage(LogType::Error, $e->getMessage()); | |
echo '{"error":{"text":"' . $e->getMessage() . '"}}'; | |
} | |
} | |
// Check if the user has a current active Subscription and determine expiration date | |
function checkSubscription($app_id, $user_id) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Checking subscription for APP ID: " . $app_id . " USER ID: " . $user_id);} | |
$result = $db->prepare("SELECT EFFECTIVE_DATE, EXPIRATION_DATE, LAST_VALIDATED FROM SUBSCRIPTIONS | |
WHERE APP_ID = '$app_id' AND USER_ID = '$user_id' LIMIT 0,1"); | |
$result->execute(); | |
$data = $result->fetch(); | |
return $data; | |
} | |
// Validate InApp Purchase Receipt, by calling the Apple iTunes verifyReceipt method | |
// *Note that this seems to take between 2-4 seconds on average | |
function verifyReceipt($receipt, $app_id, $user_id, $sandbox_override = FALSE) | |
{ | |
global $dbContainer; | |
$db = $dbContainer['db']; | |
if(isInDevelopmentMode($app_id)=="TRUE"){logMessage(LogType::Info,"Verifying receipt with Apple for APP ID: " . $app_id . " USER ID: " . $user_id);} | |
// Lookup shared secret from Publication table | |
$result = $db->query("SELECT ITUNES_SHARED_SECRET FROM PUBLICATION WHERE APP_ID = '$app_id' LIMIT 0, 1"); | |
$sharedSecret = $result->fetchColumn(); | |
if (getiTunesProductionLevel($app_id)=="sandbox" || $sandbox_override == TRUE) { | |
$endpoint = 'https://sandbox.itunes.apple.com/verifyReceipt'; | |
} | |
else { | |
$endpoint = 'https://buy.itunes.apple.com/verifyReceipt'; | |
} | |
// If no shared secret exists, don't send it to the verifyReceipt call, however it should exist! | |
if($sharedSecret){ | |
$postData = json_encode(array( | |
'receipt-data' => $receipt, | |
'password' => $sharedSecret)); | |
}else{ | |
$postData = json_encode(array( | |
'receipt-data' => $receipt)); | |
} | |
$post = $postData; | |
$context = | |
array("http"=> | |
array( | |
"method" => "POST", | |
//"header" => "custom-header: if-any\r\n" . | |
//"custom-header-two: custome-value-2\r\n" . | |
"content" => $post | |
) | |
); | |
// Create Stream Context so that it can pass in Header | |
$context = stream_context_create($context); | |
$response = file_get_contents($endpoint, false, $context); | |
if ($errno != 0) { | |
throw new Exception($errmsg, $errno); | |
} | |
$data = json_decode($response); | |
if (!is_object($data)) { | |
throw new Exception('Invalid Response Data'); | |
} | |
if (!isset($data->status) || ($data->status != 0 && $data->status != 21006)) { | |
logMessage(LogType::Warning, "Invalid receipt for APP ID: " . $app_id . " USER ID: " . $user_id . " STATUS: " . $data->status); | |
throw new Exception('Invalid Receipt', $data->status); | |
} | |
return $data; | |
} | |
/** | |
* Step 4: Run the Slim application | |
* | |
* This method should be called last. This executes the Slim application | |
* and returns the HTTP response to the HTTP client. | |
*/ | |
$app->run(); | |
// PHP doesn't support Enums so make a simple class for LogType | |
abstract class LogType | |
{ | |
const Info = 'Info'; | |
const Warning = 'Warning'; | |
const Error = 'Error'; | |
} | |
// PHP doesn't support Enums so make a simple class for AnalyticType | |
abstract class AnalyticType | |
{ | |
const Download = 'download'; | |
const ApiInteraction = 'api_interaction'; | |
} | |
// Timer class for debugging and logging use | |
class timer | |
{ | |
var $start; | |
var $pause_time; | |
/* start the timer */ | |
function timer($start = 0) | |
{ | |
if ($start) { | |
$this->start(); | |
} | |
} | |
/* start the timer */ | |
function start() | |
{ | |
$this->start = $this->get_time(); | |
$this->pause_time = 0; | |
} | |
/* pause the timer */ | |
function pause() | |
{ | |
$this->pause_time = $this->get_time(); | |
} | |
/* unpause the timer */ | |
function unpause() | |
{ | |
$this->start+= ($this->get_time() - $this->pause_time); | |
$this->pause_time = 0; | |
} | |
/* get the current timer value */ | |
function get($decimals = 8) | |
{ | |
return round(($this->get_time() - $this->start) , $decimals); | |
} | |
/* format the time in seconds */ | |
function get_time() | |
{ | |
list($usec, $sec) = explode(' ', microtime()); | |
return ((float)$usec + (float)$sec); | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment