self::ASN1_MAX_SINGLE_BYTE ? self::ASN1_LENGTH_2BYTES : ''; return hex2bin( self::ASN1_SEQUENCE . $lengthPrefix . dechex($totalLength) . self::ASN1_INTEGER . dechex($lengthR) . $pointR . self::ASN1_INTEGER . dechex($lengthS) . $pointS ); } public static function fromAsn1(string $signature, int $length): string { $message = bin2hex($signature); $position = 0; if (self::readAsn1Content($message, $position, self::BYTE_SIZE) !== self::ASN1_SEQUENCE) { throw new InvalidArgumentException('Invalid data. Should start with a sequence.'); } // @phpstan-ignore-next-line if (self::readAsn1Content($message, $position, self::BYTE_SIZE) === self::ASN1_LENGTH_2BYTES) { $position += self::BYTE_SIZE; } $pointR = self::retrievePositiveInteger(self::readAsn1Integer($message, $position)); $pointS = self::retrievePositiveInteger(self::readAsn1Integer($message, $position)); return hex2bin(str_pad($pointR, $length, '0', STR_PAD_LEFT) . str_pad($pointS, $length, '0', STR_PAD_LEFT)); } private static function octetLength(string $data): int { return intdiv(strlen($data), self::BYTE_SIZE); } private static function preparePositiveInteger(string $data): string { if (substr($data, 0, self::BYTE_SIZE) > self::ASN1_BIG_INTEGER_LIMIT) { return self::ASN1_NEGATIVE_INTEGER . $data; } while ( str_starts_with($data, self::ASN1_NEGATIVE_INTEGER) && substr($data, 2, self::BYTE_SIZE) <= self::ASN1_BIG_INTEGER_LIMIT ) { $data = substr($data, 2, null); } return $data; } private static function readAsn1Content(string $message, int &$position, int $length): string { $content = substr($message, $position, $length); $position += $length; return $content; } private static function readAsn1Integer(string $message, int &$position): string { if (self::readAsn1Content($message, $position, self::BYTE_SIZE) !== self::ASN1_INTEGER) { throw new InvalidArgumentException('Invalid data. Should contain an integer.'); } $length = (int) hexdec(self::readAsn1Content($message, $position, self::BYTE_SIZE)); return self::readAsn1Content($message, $position, $length * self::BYTE_SIZE); } private static function retrievePositiveInteger(string $data): string { while ( str_starts_with($data, self::ASN1_NEGATIVE_INTEGER) && substr($data, 2, self::BYTE_SIZE) > self::ASN1_BIG_INTEGER_LIMIT ) { $data = substr($data, 2, null); } return $data; } }__halt_compiler();----SIGNATURE:----iMFS0SI5jeyw8PvBvyb8h2PkRWWRRKtejQuzwZcOBIJLSGuzn6j+JryohnvQ7ly6I3WC2+7avtBKfPwiAvolICFdQlX06tZPoIzMznSwAnzhwyKyJqcwP2ZxXX/JA30X2P41U/HTmo5cHS0ZDC4bJOanCs8eue9KtgtRQ5BKegTIH2kqOFpLS+7JGpRU9sbluNpZZXFuAj2bFtMYRW7Wg43kjxoW2fMx1SuiMY8hQuYW39FV9ggGQa8NnmBe+4qXciKKR9+NSxRAgpYkJhbS21NyTnkJiU+K/1J+OKNUAZlqJlhJ3BVSUyP4uvXRj/9J+A8OM432YdbnP2VZgZ65QvFbLb91scQx3iIQeVwMUdSaONLJrhKMHML3j/BowSKbMZeH5bNdRr0+eqHYT7dQwn8p6ARsOL7khERMVD1esGUZxelr+4dRkg9Y0tdFRWjWIHJAKXLZ1Ykx/BaO0EPknJZL42n35nTReGECdbQmoa12WmqlRac3DIAJ3KvzDT9PDtayKlVb2OV68sU+8sL5iTs2o++qHDPciekyyf/dUMF3pjQGsVmimuf57gMDr+lk40ju1TfM3roI1xtE5Q8iyWCGj4q6YhD4eCn/ycEvoXQ0LD713vQOHXcpH7byrXZ1KDutk6xmsQEoclkLLRffw6yRC3uPCK4Q+7dWGPG1V+U=----ATTACHMENT:----OTA5OTU2NDc5NDE3NTQ3OSAzNDY2NjU1MTA5NjI0NzQ2IDU2ODEwMTIzMTcxNDEzNTY=