Skip to content

Instantly share code, notes, and snippets.

@itsbalamurali
Last active August 29, 2015 13:57
Show Gist options
  • Save itsbalamurali/9909948 to your computer and use it in GitHub Desktop.
Save itsbalamurali/9909948 to your computer and use it in GitHub Desktop.
Replace index.php
<?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