Merge branch 'MDL-76652-311' of https://github.com/lameze/moodle into MOODLE_311_STABLE
[moodle.git] / mod / lti / OAuthBody.php
blob147b695bb811b02611e0093161260953b0aa8b43
1 <?php
2 // This file is part of Moodle - http://moodle.org/
3 //
4 // Moodle is free software: you can redistribute it and/or modify
5 // it under the terms of the GNU General Public License as published by
6 // the Free Software Foundation, either version 3 of the License, or
7 // (at your option) any later version.
8 //
9 // Moodle is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 // GNU General Public License for more details.
14 // You should have received a copy of the GNU General Public License
15 // along with Moodle. If not, see <http://www.gnu.org/licenses/>.
17 // This file is part of BasicLTI4Moodle
19 // BasicLTI4Moodle is an IMS BasicLTI (Basic Learning Tools for Interoperability)
20 // consumer for Moodle 1.9 and Moodle 2.0. BasicLTI is a IMS Standard that allows web
21 // based learning tools to be easily integrated in LMS as native ones. The IMS BasicLTI
22 // specification is part of the IMS standard Common Cartridge 1.1 Sakai and other main LMS
23 // are already supporting or going to support BasicLTI. This project Implements the consumer
24 // for Moodle. Moodle is a Free Open source Learning Management System by Martin Dougiamas.
25 // BasicLTI4Moodle is a project iniciated and leaded by Ludo(Marc Alier) and Jordi Piguillem
26 // at the GESSI research group at UPC.
27 // SimpleLTI consumer for Moodle is an implementation of the early specification of LTI
28 // by Charles Severance (Dr Chuck) htp://dr-chuck.com , developed by Jordi Piguillem in a
29 // Google Summer of Code 2008 project co-mentored by Charles Severance and Marc Alier.
31 // BasicLTI4Moodle is copyright 2009 by Marc Alier Forment, Jordi Piguillem and Nikolas Galanis
32 // of the Universitat Politecnica de Catalunya http://www.upc.edu
33 // Contact info: Marc Alier Forment granludo @ gmail.com or marc.alier @ upc.edu.
35 namespace moodle\mod\lti; // Using a namespace as the basicLTI module imports classes with the same names.
37 defined('MOODLE_INTERNAL') || die;
39 require_once($CFG->dirroot . '/mod/lti/OAuth.php');
40 require_once($CFG->dirroot . '/mod/lti/TrivialStore.php');
42 /**
44 * @param int $typeid LTI type ID.
45 * @param string[] $scopes Array of scopes which give permission for the current request.
47 * @return string|int|boolean The OAuth consumer key, the LTI type ID for the validated bearer token,
48 true for requests not requiring a scope, otherwise false.
50 function get_oauth_key_from_headers($typeid = null, $scopes = null) {
51 global $DB;
53 $now = time();
55 $requestheaders = OAuthUtil::get_headers();
57 if (isset($requestheaders['Authorization'])) {
58 if (substr($requestheaders['Authorization'], 0, 6) == "OAuth ") {
59 $headerparameters = OAuthUtil::split_header($requestheaders['Authorization']);
61 return format_string($headerparameters['oauth_consumer_key']);
62 } else if (empty($scopes)) {
63 return true;
64 } else if (substr($requestheaders['Authorization'], 0, 7) == 'Bearer ') {
65 $tokenvalue = trim(substr($requestheaders['Authorization'], 7));
66 $conditions = array('token' => $tokenvalue);
67 if (!empty($typeid)) {
68 $conditions['typeid'] = intval($typeid);
70 $token = $DB->get_record('lti_access_tokens', $conditions);
71 if ($token) {
72 // Log token access.
73 $DB->set_field('lti_access_tokens', 'lastaccess', $now, array('id' => $token->id));
74 $permittedscopes = json_decode($token->scope);
75 if ((intval($token->validuntil) > $now) && !empty(array_intersect($scopes, $permittedscopes))) {
76 return intval($token->typeid);
81 return false;
84 function handle_oauth_body_post($oauthconsumerkey, $oauthconsumersecret, $body, $requestheaders = null) {
86 if ($requestheaders == null) {
87 $requestheaders = OAuthUtil::get_headers();
90 // Must reject application/x-www-form-urlencoded.
91 if (isset($requestheaders['Content-type'])) {
92 if ($requestheaders['Content-type'] == 'application/x-www-form-urlencoded' ) {
93 throw new OAuthException("OAuth request body signing must not use application/x-www-form-urlencoded");
97 if (isset($requestheaders['Authorization']) && (substr($requestheaders['Authorization'], 0, 6) == "OAuth ")) {
98 $headerparameters = OAuthUtil::split_header($requestheaders['Authorization']);
99 $oauthbodyhash = $headerparameters['oauth_body_hash'];
102 if ( ! isset($oauthbodyhash) ) {
103 throw new OAuthException("OAuth request body signing requires oauth_body_hash body");
106 // Verify the message signature.
107 $store = new TrivialOAuthDataStore();
108 $store->add_consumer($oauthconsumerkey, $oauthconsumersecret);
110 $server = new OAuthServer($store);
112 $method = new OAuthSignatureMethod_HMAC_SHA1();
113 $server->add_signature_method($method);
114 $request = OAuthRequest::from_request();
116 try {
117 $server->verify_request($request);
118 } catch (\Exception $e) {
119 $message = $e->getMessage();
120 throw new OAuthException("OAuth signature failed: " . $message);
123 $postdata = $body;
125 $hash = base64_encode(sha1($postdata, true));
127 if ( $hash != $oauthbodyhash ) {
128 throw new OAuthException("OAuth oauth_body_hash mismatch");
131 return $postdata;