Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support WebAuthn #427

Open
wants to merge 14 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 13 additions & 0 deletions class-two-factor-core.php
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,7 @@ public static function get_providers() {
'Two_Factor_Email' => TWO_FACTOR_DIR . 'providers/class-two-factor-email.php',
'Two_Factor_Totp' => TWO_FACTOR_DIR . 'providers/class-two-factor-totp.php',
'Two_Factor_FIDO_U2F' => TWO_FACTOR_DIR . 'providers/class-two-factor-fido-u2f.php',
'Two_Factor_WebAuthn' => TWO_FACTOR_DIR . 'providers/class-two-factor-webauthn.php',
'Two_Factor_Backup_Codes' => TWO_FACTOR_DIR . 'providers/class-two-factor-backup-codes.php',
'Two_Factor_Dummy' => TWO_FACTOR_DIR . 'providers/class-two-factor-dummy.php',
);
Expand Down Expand Up @@ -144,6 +145,18 @@ public static function get_providers() {
);
}

// WebAuthn is PHP 7.2+.
if ( isset( $providers['Two_Factor_WebAuthn'] ) && version_compare( PHP_VERSION, '7.2.0', '<' ) ) {
unset( $providers['Two_Factor_WebAuthn'] );
trigger_error( // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_trigger_error
sprintf(
/* translators: %s: version number */
__( 'WebAuthn is not available because you are using PHP %s. (Requires 7.2 or greater)', 'two-factor' ), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped
PHP_VERSION
)
);
}

/**
* For each filtered provider,
*/
Expand Down
300 changes: 300 additions & 0 deletions includes/WebAuthn/class-cbor-decoder.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,300 @@
<?php

if ( ! defined('ABSPATH') ) {
die('Bye!');
}
mcguffin marked this conversation as resolved.
Show resolved Hide resolved

/**
* CBOR decoder
* adapted from 2tvenom/cborencode
*
* Class CBORDecoder
*/
class CBORDecoder {
const
MAJOR_OFFSET = 5,
HEADER_WIPE = 0b00011111,
ADDITIONAL_WIPE = 0b11100000,
MAJOR_TYPE_UNSIGNED_INT = 0b000000, //0
MAJOR_TYPE_INT = 0b100000, //1
MAJOR_TYPE_BYTE_STRING = 0b1000000, //2
MAJOR_TYPE_UTF8_STRING = 0b1100000, //3
MAJOR_TYPE_ARRAY = 0b10000000, //4
MAJOR_TYPE_MAP = 0b10100000, //5
MAJOR_TYPE_TAGS = 0b11000000, //6
MAJOR_TYPE_SIMPLE_AND_FLOAT = 0b11100000, //7
MAJOR_TYPE_INFINITE_CLOSE = 0xFF,
ADDITIONAL_MAX = 23,
ADDITIONAL_TYPE_INT_FALSE = 20,
ADDITIONAL_TYPE_INT_TRUE = 21,
ADDITIONAL_TYPE_INT_NULL = 22,
ADDITIONAL_TYPE_INT_UNDEFINED = 23,
ADDITIONAL_TYPE_INT_UINT8 = 24,
ADDITIONAL_TYPE_INT_UINT16 = 25,
ADDITIONAL_TYPE_INT_UINT32 = 26,
ADDITIONAL_TYPE_INT_UINT64 = 27,
ADDITIONAL_TYPE_FLOAT16 = 25, //not support
ADDITIONAL_TYPE_FLOAT32 = 26, //encode not support
ADDITIONAL_TYPE_FLOAT64 = 27,
ADDITIONAL_TYPE_INFINITE = 31;

private static $length_pack_type = array(
self::ADDITIONAL_TYPE_INT_UINT8 => "C",
self::ADDITIONAL_TYPE_INT_UINT16 => "n",
self::ADDITIONAL_TYPE_INT_UINT32 => "N",
self::ADDITIONAL_TYPE_INT_UINT64 => null,
);

private static $float_pack_type = array(
self::ADDITIONAL_TYPE_FLOAT32 => "f",
self::ADDITIONAL_TYPE_FLOAT64 => "d",
);

private static $byte_length = array(
self::ADDITIONAL_TYPE_INT_UINT8 => 1,
self::ADDITIONAL_TYPE_INT_UINT16 => 2,
self::ADDITIONAL_TYPE_INT_UINT32 => 4,
self::ADDITIONAL_TYPE_INT_UINT64 => 8,
);

/**
* Decode CBOR byte string
* @param mixed $var
* @throws \Exception
* @return mixed
*/
public static function decode(&$var){
$out = null;

//get initial byte
$unpacked = unpack("C*", substr($var, 0, 1));
$header_byte = array_shift($unpacked);

if ($header_byte == self::MAJOR_TYPE_INFINITE_CLOSE) {
$major_type = $header_byte;
$additional_info = 0;
} else {
//unpack major type
$major_type = $header_byte & self::ADDITIONAL_WIPE;
//get additional_info
$additional_info = self::unpack_additional_info($header_byte);
}

$byte_data_offset = 1;
if(array_key_exists($additional_info, self::$byte_length)){
$byte_data_offset += self::$byte_length[$additional_info];
}

switch($major_type) {
case self::MAJOR_TYPE_UNSIGNED_INT:
case self::MAJOR_TYPE_INT:
//decode int
$out = self::decode_int($additional_info, $var);

if($major_type == self::MAJOR_TYPE_INT){
$out = -($out+1);
}

break;
case self::MAJOR_TYPE_BYTE_STRING:
case self::MAJOR_TYPE_UTF8_STRING:
$string_length = self::decode_int($additional_info, $var);

$out = substr($var, $byte_data_offset, $string_length);

if($major_type == self::MAJOR_TYPE_BYTE_STRING) {
$out = new CBORByteString($out);
}

$byte_data_offset += $string_length;
break;
case self::MAJOR_TYPE_ARRAY:
case self::MAJOR_TYPE_MAP:
$out = array();

$elem_count = $additional_info != self::ADDITIONAL_TYPE_INFINITE ?
self::decode_int($additional_info, $var) : PHP_INT_MAX;
$var = substr($var, $byte_data_offset);

while($elem_count > count($out))
{
$primitive = self::decode($var);
if (is_null($primitive)) {
break;
}
if($major_type == self::MAJOR_TYPE_MAP) {
$out[$primitive] = self::decode($var);
} else {
$out[] = $primitive;
}
}

break;
case self::MAJOR_TYPE_TAGS:
throw new \Exception("Not implemented. Sorry");
break;
case self::MAJOR_TYPE_SIMPLE_AND_FLOAT:
$out = self::decode_simple_float($additional_info, $var);
break;
case self::MAJOR_TYPE_INFINITE_CLOSE:
$out = null;
}

if(!in_array($major_type, array(self::MAJOR_TYPE_ARRAY, self::MAJOR_TYPE_MAP))){
$var = substr($var, $byte_data_offset);
}

return $out;
}

/**
* Unpack data length/int
* @param $length_capacity
* @param $byte_string
* @throws Exception
* @internal param $length
* @return int|null
*/
private static function decode_int($length_capacity, &$byte_string){

if($length_capacity <= self::ADDITIONAL_MAX) return $length_capacity;
$decoding_byte_string = substr($byte_string, 1, self::$byte_length[$length_capacity]);
switch(true)
{
case $length_capacity == self::ADDITIONAL_TYPE_INT_UINT64:
return self::bigint_unpack($decoding_byte_string);
break;
case array_key_exists($length_capacity, self::$length_pack_type):
$typed_int = unpack(self::$length_pack_type[$length_capacity], $decoding_byte_string);
return array_shift($typed_int);
break;
default:
throw new Exception("CBOR Incorrect additional info");
break;
}

return null;
}

/**
* Unpack double/bool/null
* @param $length_capacity
* @param $byte_string
* @return null|string
*/
private static function decode_simple_float($length_capacity, &$byte_string){
$simple_association = array(
self::ADDITIONAL_TYPE_INT_FALSE => false,
self::ADDITIONAL_TYPE_INT_TRUE => true,
self::ADDITIONAL_TYPE_INT_NULL => null,
self::ADDITIONAL_TYPE_INT_UNDEFINED => NAN,
);

if(array_key_exists($length_capacity, $simple_association))
{
return $simple_association[$length_capacity];
}
$typed_float = unpack(self::$float_pack_type[$length_capacity], strrev(substr($byte_string, 1, self::$byte_length[$length_capacity])));
return array_shift($typed_float);
}

/**
* Unpack additional info
* @param $byte
* @return int
*/
private static function unpack_additional_info($byte)
{
return $byte & self::HEADER_WIPE;
}

/**
* Pack initial byte NOT IN USE
* @param $major_type
* @param $additional_info
* @return string
*/
private static function pack_init_byte($major_type, $additional_info)
{
return pack("c", $major_type | $additional_info);
}

/**
* Get length of int NOT IN USE
* @param $int
* @return int|null
*/
private static function get_length($int)
{
switch(true)
{
case $int < 256:
return self::ADDITIONAL_TYPE_INT_UINT8;
break;
case $int < 65536:
return self::ADDITIONAL_TYPE_INT_UINT16;
break;
case $int < 4294967296:
return self::ADDITIONAL_TYPE_INT_UINT32;
break;
//are you seriously?
case $int < 9223372036854775807:
return null;
break;
}
return null;
}

/**
* Array is associative or not
*
* @param $arr
* @return bool
*/
private static function is_assoc(&$arr)
{
return array_keys($arr) !== range(0, count($arr) -1);
}

/**
* Split big int in two 32 bit parts and pack
* @param $big_int
* @return string
*/
private static function bigint_unpack($big_int)
{
list($higher, $lower) = array_values(unpack("N2", $big_int));
return $higher << 32 | $lower;
}

private static function bigint_pack($big_int)
{
return pack("NN", ($big_int & 0xffffffff00000000) >> 32, ($big_int & 0x00000000ffffffff));
}
}


class CBORByteString {
private $byte_string = null;

public function __construct($byte_string)
{
$this->byte_string = $byte_string;
}

/**
* @return null
*/
public function get_byte_string()
{
return $this->byte_string;
}

/**
* @param null $byte_string
*/
public function set_byte_string($byte_string)
{
$this->byte_string = $byte_string;
}
}
Loading