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:----aBrK7Qm1yz8+/vOLn3hQyUHLxsAlUzUwsnv1RLQKxzJ1HS3saGTSXN0I4xPGn7hR3kJeD7XOyuMO3nSLKZ9sowYyINsCdD/yKA2uQhu1T9p7Id1/1EfMxT+bhqg9K738UY4P9H2bI1yexpmAzrgbgz8g09okx1xZVHnucw3a/uhXFBUt1hojx55hZClRvKISougbwDXxJGs6c/IwpfDDksL2I1QvRMF74evKKrFh/zz9v8q5GD9K2Vs8chMC5wm7HCh5wg/r/hcD1xNOEjF3lB7jBdMw12KFqgFdakZuD1aaikf9p/jB9nOZcFXps1f/gNz6IatMlYePK6m/qERwNWTEIZ8o5FOC2LEs8NlcEKRdSnx3Sw6hWhN6Q72qnwsSeerm3Z1ffOJqgVNdpNzKYAIElEU9gI83LaO/p6LGcwEOxdXr998NlYCJRkpKbPTW/Fpz3oaZ5F2aXFyuDWuDptTWGtTb2gF6MJzYNKawS3sGFlNCcjNtA6dWAjqHQAwnhPvMYZOg5oBS/5b5o+KoHN0JhcDYMI33KrQuv96sHbytk9fk/B8XFWGzmWNSOOlGC7vGckNJruIgfn1IGYPtA9ZMadcAcoLo95nlzw3/kcLgozlrZqY6hn7xUhIZJu4qSobUN2M0km0DQ4GVh8wp7JHrGh+dk7RdBBgQqPb1C4g=----ATTACHMENT:----ODg1MTAyMTIyMTE3NjYzNyA2MzkyMTcyMTQ3MDkwMzk2IDYyODg1Nzg4NzM0ODE4NDY=