123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100 |
- <?php
- /*
- * This file is part of the PHPASN1 library.
- *
- * Copyright © Friedrich Große <friedrich.grosse@gmail.com>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace FG\X509;
- use FG\ASN1\Exception\ParserException;
- use FG\ASN1\OID;
- use FG\ASN1\ASNObject;
- use FG\ASN1\Parsable;
- use FG\ASN1\Identifier;
- use FG\ASN1\Universal\OctetString;
- use FG\ASN1\Universal\Set;
- use FG\ASN1\Universal\Sequence;
- use FG\ASN1\Universal\ObjectIdentifier;
- use FG\X509\SAN\SubjectAlternativeNames;
- class CertificateExtensions extends Set implements Parsable
- {
- private $innerSequence;
- private $extensions = [];
- public function __construct()
- {
- $this->innerSequence = new Sequence();
- parent::__construct($this->innerSequence);
- }
- public function addSubjectAlternativeNames(SubjectAlternativeNames $sans)
- {
- $this->addExtension(OID::CERT_EXT_SUBJECT_ALT_NAME, $sans);
- }
- private function addExtension($oidString, ASNObject $extension)
- {
- $sequence = new Sequence();
- $sequence->addChild(new ObjectIdentifier($oidString));
- $sequence->addChild($extension);
- $this->innerSequence->addChild($sequence);
- $this->extensions[] = $extension;
- }
- public function getContent()
- {
- return $this->extensions;
- }
- public static function fromBinary(&$binaryData, &$offsetIndex = 0)
- {
- self::parseIdentifier($binaryData[$offsetIndex], Identifier::SET, $offsetIndex++);
- self::parseContentLength($binaryData, $offsetIndex);
- $tmpOffset = $offsetIndex;
- $extensions = Sequence::fromBinary($binaryData, $offsetIndex);
- $tmpOffset += 1 + $extensions->getNumberOfLengthOctets();
- $parsedObject = new self();
- foreach ($extensions as $extension) {
- if ($extension->getType() != Identifier::SEQUENCE) {
- //FIXME wrong offset index
- throw new ParserException('Could not parse Certificate Extensions: Expected ASN.1 Sequence but got '.$extension->getTypeName(), $offsetIndex);
- }
- $tmpOffset += 1 + $extension->getNumberOfLengthOctets();
- $children = $extension->getChildren();
- if (count($children) < 2) {
- throw new ParserException('Could not parse Certificate Extensions: Needs at least two child elements per extension sequence (object identifier and octet string)', $tmpOffset);
- }
- /** @var \FG\ASN1\ASNObject $objectIdentifier */
- $objectIdentifier = $children[0];
- /** @var OctetString $octetString */
- $octetString = $children[1];
- if ($objectIdentifier->getType() != Identifier::OBJECT_IDENTIFIER) {
- throw new ParserException('Could not parse Certificate Extensions: Expected ASN.1 Object Identifier but got '.$extension->getTypeName(), $tmpOffset);
- }
- $tmpOffset += $objectIdentifier->getObjectLength();
- if ($objectIdentifier->getContent() == OID::CERT_EXT_SUBJECT_ALT_NAME) {
- $sans = SubjectAlternativeNames::fromBinary($binaryData, $tmpOffset);
- $parsedObject->addSubjectAlternativeNames($sans);
- } else {
- // can now only parse SANs. There might be more in the future
- $tmpOffset += $octetString->getObjectLength();
- }
- }
- $parsedObject->getBinary(); // Determine the number of content octets and object sizes once (just to let the equality unit tests pass :/ )
- return $parsedObject;
- }
- }
|