|
|
- <?php
- use \Firebase\JWT\JWT;
-
- $app->get('/alexa', function() use($app) {
- render('alexa', array(
- 'title' => 'Teacup for Alexa'
- ));
- });
-
- $app->get('/alexa/auth', function() use($app) {
- $req = $app->request();
- $params = $req->params();
-
- $required = ['client_id', 'response_type', 'state', 'redirect_uri'];
- $params_present = array_keys($params);
-
- // Validate Alexa OAuth parameters
- if(count(array_intersect($required, $params_present)) != count($required)) {
- render('auth_error', array(
- 'title' => 'Sign In',
- 'error' => 'Missing parameters',
- 'errorDescription' => 'One or more required parameters were missing',
- 'footer' => false
- ));
- return;
- }
-
- // Check that redirect URI is one that is allowed
- if(!in_array($params['redirect_uri'], Config::$alexaRedirectURIs)) {
- render('auth_error', array(
- 'title' => 'Sign In',
- 'error' => 'Invalid redirect URI',
- 'errorDescription' => 'Alexa sent an invalid redirect URI',
- 'footer' => false
- ));
- return;
- }
-
- if($params['client_id'] != Config::$alexaClientID) {
- render('auth_error', array(
- 'title' => 'Sign In',
- 'error' => 'Invalid Client ID',
- 'errorDescription' => 'Alexa sent an invalid client ID',
- 'footer' => false
- ));
- return;
- }
-
- // Pass through the OAuth parameters
- render('alexa-auth', [
- 'title' => 'Teacup for Alexa',
- 'client_id' => $params['client_id'],
- 'response_type' => $params['response_type'],
- 'state' => $params['state'],
- 'redirect_uri' => $params['redirect_uri'],
- 'footer' => false
- ]);
- });
-
- $app->post('/alexa/login', function() use($app) {
- $req = $app->request();
- $params = $req->params();
-
- $required = ['code', 'client_id', 'state', 'redirect_uri'];
- $params_present = array_keys($params);
-
- if(count(array_intersect($required, $params_present)) != count($required)) {
- render('auth_error', array(
- 'title' => 'Sign In',
- 'error' => 'Missing parameters',
- 'errorDescription' => 'One or more required parameters were missing',
- 'footer' => false
- ));
- return;
- }
-
- $user = ORM::for_table('users')
- ->where('device_code', $params['code'])
- ->where_gt('device_code_expires', date('Y-m-d H:i:s'))->find_one();
-
- if(!$user) {
- render('auth_error', array(
- 'title' => 'Sign In',
- 'error' => 'Invalid code',
- 'errorDescription' => 'The code you entered is invalid or has expired',
- 'footer' => false
- ));
- return;
- }
-
- $code = JWT::encode(array(
- 'user_id' => $user->id,
- 'iat' => time(),
- 'exp' => time()+300,
- 'client_id' => $params['client_id'],
- 'state' => $params['state'],
- 'redirect_uri' => $params['redirect_uri'],
- ), Config::$jwtSecret);
-
- $redirect = $params['redirect_uri'] . '?code=' . $code . '&state=' . $params['state'];
-
- $app->redirect($redirect, 302);
- });
-
- $app->post('/alexa/token', function() use($app) {
- $req = $app->request();
- $params = $req->params();
- // Alexa requests a token given a code generated above
-
- // Verify the client ID and secret
- if($params['client_id'] != Config::$alexaClientID
- || $params['client_secret'] != Config::$alexaClientSecret) {
- $app->response->setStatus(400);
- $app->response()['Content-type'] = 'application/json';
- $app->response()->body(json_encode([
- 'error' => 'forbidden',
- 'error_description' => 'The client ID and secret do not match'
- ]));
- return;
- }
-
- if(array_key_exists('code', $params)) {
- $jwt = $params['code'];
- } elseif(array_key_exists('refresh_token', $params)) {
- $jwt = $params['refresh_token'];
- } else {
- $app->response->setStatus(400);
- $app->response()['Content-type'] = 'application/json';
- $app->response()->body(json_encode([
- 'error' => 'bad_request',
- 'error_description' => 'Must provide either an authorization code or refresh token'
- ]));
- return;
- }
-
- // Validate the JWT
- try {
- $user = JWT::decode($jwt, Config::$jwtSecret, ['HS256']);
- } catch(Exception $e) {
- $app->response->setStatus(400);
- $app->response()['Content-type'] = 'application/json';
- $app->response()->body(json_encode([
- 'error' => 'unauthorized',
- 'error_description' => 'The authorization code or refresh token was invalid'
- ]));
- return;
- }
-
- // Generate an access token and refresh token
- $access_token = JWT::encode([
- 'user_id' => $user->user_id,
- 'client_id' => $user->client_id,
- 'iat' => time(),
- ], Config::$jwtSecret);
- $refresh_token = JWT::encode([
- 'user_id' => $user->user_id,
- 'client_id' => $user->client_id,
- 'iat' => time(),
- ], Config::$jwtSecret);
-
-
- $app->response()['Content-type'] = 'application/json';
- $app->response()->body(json_encode([
- 'access_token' => $access_token,
- 'refresh_token' => $refresh_token
- ]));
- });
-
-
- $app->post('/alexa/endpoint', function() use($app) {
-
- $input = file_get_contents('php://input');
- $json = json_decode($input, 'input');
-
- $alexaRequest = \Alexa\Request\Request::fromData($json);
-
- if($alexaRequest instanceof Alexa\Request\IntentRequest) {
-
- # Verify the access token
- try {
- $data = JWT::decode($alexaRequest->user->accessToken, Config::$jwtSecret, ['HS256']);
- } catch(Exception $e) {
- $app->response->setStatus(401);
- $app->response()['Content-type'] = 'application/json';
- $app->response()->body(json_encode([
- 'error' => 'unauthorized',
- 'error_description' => 'The access token was invalid or has expired'
- ]));
- return;
- }
-
- $user = ORM::for_table('users')->find_one($data->user_id);
-
- if(!$user) {
- $app->response->setStatus(400);
- return;
- }
-
- $action = $alexaRequest->slots['Action'];
- $action = ($action == 'drank' ? 'drank' : 'ate');
- $food = ucfirst($alexaRequest->slots['Food']);
-
-
- $entry = ORM::for_table('entries')->create();
- $entry->user_id = $user->id;
- $entry->type = ($action == 'drank' ? 'drink' : 'eat');
- $entry->content = $food;
- $entry->published = date('Y-m-d H:i:s');
- $entry->save();
-
- $text_content = 'Just ' . $action . ': ' . $food;
-
- if($user->micropub_endpoint) {
- $mp_request = array(
- 'h' => 'entry',
- 'published' => date('Y-m-d H:i:s'),
- 'summary' => $text_content
- );
- if($user->enable_array_micropub) {
- $mp_request[$action] = [
- 'type' => 'h-food',
- 'properties' => [
- 'name' => $food
- ]
- ];
- } else {
- $mp_request['p3k-food'] = $food;
- $mp_request['p3k-type'] = $entry->type;
- }
-
- $r = micropub_post($user->micropub_endpoint, $mp_request, $user->access_token);
- $request = $r['request'];
- $response = $r['response'];
-
- $entry->micropub_response = $response;
- if($response && preg_match('/Location: (.+)/', $response, $match)) {
- $url = $match[1];
- $entry->micropub_success = 1;
- $entry->canonical_url = $url;
- } else {
- $entry->micropub_success = 0;
- $url = Config::$base_url . $user->url . '/' . $entry->id;
- }
- $entry->save();
- }
-
-
- $response = new \Alexa\Response\Response;
- $response->respond('Got it!')
- ->withCard('You '.$action.': '.$food, $url)
- ->endSession(true);
-
- $app->response()['Content-type'] = 'application/json';
- $app->response()->body(json_encode($response->render()));
- }
- });
|