2016-09-05 21:21:36 +01:00
< ? php
/**
* Pterodactyl - Panel
2016-12-07 22:46:38 +00:00
* Copyright ( c ) 2015 - 2016 Dane Everitt < dane @ daneeveritt . com >.
2016-09-05 21:21:36 +01:00
*
* Permission is hereby granted , free of charge , to any person obtaining a copy
* of this software and associated documentation files ( the " Software " ), to deal
* in the Software without restriction , including without limitation the rights
* to use , copy , modify , merge , publish , distribute , sublicense , and / or sell
* copies of the Software , and to permit persons to whom the Software is
* furnished to do so , subject to the following conditions :
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
* IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER
* LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM ,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE .
*/
2016-12-07 22:46:38 +00:00
2016-09-05 21:21:36 +01:00
namespace Pterodactyl\Http\Middleware ;
2016-10-15 01:22:23 +01:00
use Auth ;
2016-09-05 21:21:36 +01:00
use Crypt ;
2016-10-15 01:22:23 +01:00
use Config ;
2016-09-05 21:21:36 +01:00
use IPTools\IP ;
use IPTools\Range ;
2016-12-07 22:46:38 +00:00
use Dingo\Api\Routing\Route ;
use Illuminate\Http\Request ;
use Pterodactyl\Models\User ;
2016-09-05 21:21:36 +01:00
use Pterodactyl\Models\APIKey ;
use Pterodactyl\Models\APIPermission ;
2016-10-07 21:06:09 +01:00
use Pterodactyl\Services\APILogService ;
2016-09-05 21:21:36 +01:00
use Dingo\Api\Auth\Provider\Authorization ;
2016-12-07 22:46:38 +00:00
use Symfony\Component\HttpKernel\Exception\HttpException ; // 400
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException ; // 401
2016-09-05 21:21:36 +01:00
use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException ; // 403
2016-12-07 22:46:38 +00:00
use Symfony\Component\HttpKernel\Exception\UnauthorizedHttpException ; //500
2016-09-05 21:21:36 +01:00
class APISecretToken extends Authorization
{
protected $algo = 'sha256' ;
protected $permissionAllowed = false ;
protected $url = '' ;
public function __construct ()
{
2016-10-15 01:22:23 +01:00
Config :: set ( 'session.driver' , 'array' );
2016-09-05 21:21:36 +01:00
}
public function getAuthorizationMethod ()
{
return 'Authorization' ;
}
public function authenticate ( Request $request , Route $route )
{
2016-12-07 22:46:38 +00:00
if ( ! $request -> bearerToken () || empty ( $request -> bearerToken ())) {
2016-10-07 21:10:54 +01:00
APILogService :: log ( $request , 'The authentication header was missing or malformed.' );
throw new UnauthorizedHttpException ( 'The authentication header was missing or malformed.' );
2016-09-05 21:21:36 +01:00
}
list ( $public , $hashed ) = explode ( '.' , $request -> bearerToken ());
$key = APIKey :: where ( 'public' , $public ) -> first ();
2016-12-07 22:46:38 +00:00
if ( ! $key ) {
2016-10-07 21:10:54 +01:00
APILogService :: log ( $request , 'Invalid API Key.' );
2016-09-05 21:21:36 +01:00
throw new AccessDeniedHttpException ( 'Invalid API Key.' );
}
// Check for Resource Permissions
2016-12-07 22:46:38 +00:00
if ( ! empty ( $request -> route () -> getName ())) {
if ( ! is_null ( $key -> allowed_ips )) {
2016-09-05 21:21:36 +01:00
$inRange = false ;
2016-12-07 22:46:38 +00:00
foreach ( json_decode ( $key -> allowed_ips ) as $ip ) {
2016-09-05 21:21:36 +01:00
if ( Range :: parse ( $ip ) -> contains ( new IP ( $request -> ip ()))) {
$inRange = true ;
break ;
}
}
2016-12-07 22:46:38 +00:00
if ( ! $inRange ) {
2016-10-07 21:10:54 +01:00
APILogService :: log ( $request , 'This IP address <' . $request -> ip () . '> does not have permission to use this API key.' );
2016-09-05 21:21:36 +01:00
throw new AccessDeniedHttpException ( 'This IP address <' . $request -> ip () . '> does not have permission to use this API key.' );
}
}
2016-10-20 23:35:55 +01:00
$permission = APIPermission :: where ( 'key_id' , $key -> id ) -> where ( 'permission' , $request -> route () -> getName ());
// Suport Wildcards
if ( starts_with ( $request -> route () -> getName (), 'api.user' )) {
$permission -> orWhere ( 'permission' , 'api.user.*' );
2016-12-07 22:46:38 +00:00
} elseif ( starts_with ( $request -> route () -> getName (), 'api.admin' )) {
2016-10-20 23:35:55 +01:00
$permission -> orWhere ( 'permission' , 'api.admin.*' );
}
2016-12-07 22:46:38 +00:00
if ( ! $permission -> first ()) {
2016-10-20 23:35:55 +01:00
APILogService :: log ( $request , 'You do not have permission to access this resource. This API Key requires the ' . $request -> route () -> getName () . ' permission node.' );
throw new AccessDeniedHttpException ( 'You do not have permission to access this resource. This API Key requires the ' . $request -> route () -> getName () . ' permission node.' );
2016-09-05 21:21:36 +01:00
}
}
try {
$decrypted = Crypt :: decrypt ( $key -> secret );
} catch ( \Illuminate\Contracts\Encryption\DecryptException $ex ) {
2016-10-07 21:10:54 +01:00
APILogService :: log ( $request , 'There was an error while attempting to check your secret key.' );
2016-09-05 21:21:36 +01:00
throw new HttpException ( 'There was an error while attempting to check your secret key.' );
}
$this -> url = urldecode ( $request -> fullUrl ());
2016-12-07 22:46:38 +00:00
if ( $this -> _generateHMAC ( $request -> getContent (), $decrypted ) !== base64_decode ( $hashed )) {
2016-10-07 21:10:54 +01:00
APILogService :: log ( $request , 'The hashed body was not valid. Potential modification of contents in route.' );
2016-09-05 21:21:36 +01:00
throw new BadRequestHttpException ( 'The hashed body was not valid. Potential modification of contents in route.' );
}
2016-10-07 21:06:09 +01:00
// Log the Route Access
2016-10-07 21:10:54 +01:00
APILogService :: log ( $request , null , true );
2016-09-05 21:21:36 +01:00
2016-12-07 22:46:38 +00:00
return Auth :: loginUsingId ( $key -> user );
2016-09-05 21:21:36 +01:00
}
protected function _generateHMAC ( $body , $key )
{
$data = $this -> url . $body ;
2016-12-07 22:46:38 +00:00
2016-09-05 21:21:36 +01:00
return hash_hmac ( $this -> algo , $data , $key , true );
}
}