ASDecoder.php 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. <?php
  2. namespace AppleSignIn;
  3. use AppleSignIn\Vendor\JWK;
  4. use AppleSignIn\Vendor\JWT;
  5. use Exception;
  6. class ASDecoder
  7. {
  8. /**
  9. * Parse a provided Sign In with Apple identity token.
  10. * @param $identityToken
  11. * @return ASPayload
  12. * @throws Exception
  13. */
  14. public static function getAppleSignInPayload($identityToken)
  15. {
  16. $identityPayload = self::decodeIdentityToken($identityToken);
  17. return new ASPayload($identityPayload);
  18. }
  19. /**
  20. * Decode the Apple encoded JWT using Apple's public key for the signing.
  21. * @param $identityToken
  22. * @return object
  23. * @throws Exception
  24. */
  25. public static function decodeIdentityToken($identityToken)
  26. {
  27. $publicKeyData = self::fetchPublicKey($identityToken);
  28. $publicKey = $publicKeyData['publicKey'];
  29. $alg = $publicKeyData['alg'];
  30. $payload = JWT::decode($identityToken, $publicKey, [$alg]);
  31. return $payload;
  32. }
  33. /**
  34. * Fetch Apple's public key from the auth/keys REST API to use to decode
  35. * the Sign In JWT.
  36. *
  37. * @param $identityToken
  38. * @return array
  39. * @throws Exception
  40. */
  41. public static function fetchPublicKey($identityToken)
  42. {
  43. $publicKeys = file_get_contents('https://appleid.apple.com/auth/keys');
  44. $decodedPublicKeys = json_decode($publicKeys, true);
  45. if (!isset($decodedPublicKeys['keys']) || count($decodedPublicKeys['keys']) < 1) {
  46. throw new Exception('Invalid key format.');
  47. }
  48. // 苹果公钥返回的 keys 内数据不是固定顺序,此处按索引取 auth keys,取正确的 key
  49. // value by index
  50. try {
  51. $tks = explode('.', $identityToken);
  52. if (count($tks) != 3) {
  53. throw new Exception('Wrong number of segments');
  54. }
  55. list($headb64, $bodyb64, $cryptob64) = $tks;
  56. $header = JWT::jsonDecode(JWT::urlsafeB64Decode($headb64));
  57. $kid = $header->kid;
  58. if (count($decodedPublicKeys['keys']) > 1) {
  59. $indexPublicInfo = array_column($decodedPublicKeys['keys'], null, 'kid');
  60. $parsedKeyData = isset($indexPublicInfo[$kid]) ? $indexPublicInfo[$kid] : $decodedPublicKeys['keys'][0];
  61. } else {
  62. $parsedKeyData = $decodedPublicKeys['keys'][0];
  63. }
  64. } catch (\Exception $exception) {
  65. $parsedKeyData = $decodedPublicKeys['keys'][0];
  66. }
  67. $parsedPublicKey = JWK::parseKey($parsedKeyData);
  68. $publicKeyDetails = openssl_pkey_get_details($parsedPublicKey);
  69. if (!isset($publicKeyDetails['key'])) {
  70. throw new Exception('Invalid public key details.');
  71. }
  72. return [
  73. 'publicKey' => $publicKeyDetails['key'],
  74. 'alg' => $parsedKeyData['alg']
  75. ];
  76. }
  77. }
  78. /**
  79. * A class decorator for the Sign In with Apple payload produced by
  80. * decoding the signed JWT from a client.
  81. */
  82. class ASPayload
  83. {
  84. protected $_instance;
  85. public function __construct($instance)
  86. {
  87. if (is_null($instance)) {
  88. throw new Exception('ASPayload received null instance.');
  89. }
  90. $this->_instance = $instance;
  91. }
  92. public function __call($method, $args)
  93. {
  94. return call_user_func_array(array($this->_instance, $method), $args);
  95. }
  96. public function __get($key)
  97. {
  98. return $this->_instance->$key;
  99. }
  100. public function __set($key, $val)
  101. {
  102. return $this->_instance->$key = $val;
  103. }
  104. public function getEmail()
  105. {
  106. return (isset($this->_instance->email)) ? $this->_instance->email : null;
  107. }
  108. public function getUser()
  109. {
  110. return (isset($this->_instance->sub)) ? $this->_instance->sub : null;
  111. }
  112. public function verifyUser($user)
  113. {
  114. return $user === $this->getUser();
  115. }
  116. }