OLD | NEW |
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 part of dart.crypto; | 5 part of dart.crypto; |
6 | 6 |
7 class _HMAC implements HMAC { | 7 class _HMAC implements HMAC { |
| 8 bool _isClosed = false; |
| 9 |
8 _HMAC(Hash this._hash, List<int> this._key) : _message = []; | 10 _HMAC(Hash this._hash, List<int> this._key) : _message = []; |
9 | 11 |
10 HMAC update(List<int> data) { | 12 add(List<int> data) { |
| 13 if (_isClosed) throw new StateError("HMAC is closed"); |
11 _message.addAll(data); | 14 _message.addAll(data); |
12 return this; | |
13 } | 15 } |
14 | 16 |
15 List<int> digest() { | 17 List<int> get digest { |
16 var blockSize = _hash.blockSize; | 18 var blockSize = _hash.blockSize; |
17 | 19 |
18 // Hash the key if it is longer than the block size of the hash. | 20 // Hash the key if it is longer than the block size of the hash. |
19 if (_key.length > blockSize) { | 21 if (_key.length > blockSize) { |
20 _hash = _hash.newInstance(); | 22 _hash = _hash.newInstance(); |
21 _key = _hash.update(_key).digest(); | 23 _hash.add(_key); |
| 24 _key = _hash.close(); |
22 } | 25 } |
23 | 26 |
24 // Zero-pad the key until its size is equal to the block size of the hash. | 27 // Zero-pad the key until its size is equal to the block size of the hash. |
25 if (_key.length < blockSize) { | 28 if (_key.length < blockSize) { |
26 var newKey = new List(blockSize); | 29 var newKey = new List.fixedLength(blockSize); |
27 newKey.setRange(0, _key.length, _key); | 30 newKey.setRange(0, _key.length, _key); |
28 for (var i = _key.length; i < blockSize; i++) { | 31 for (var i = _key.length; i < blockSize; i++) { |
29 newKey[i] = 0; | 32 newKey[i] = 0; |
30 } | 33 } |
31 _key = newKey; | 34 _key = newKey; |
32 } | 35 } |
33 | 36 |
34 // Compute inner padding. | 37 // Compute inner padding. |
35 var padding = new List(blockSize); | 38 var padding = new List.fixedLength(blockSize); |
36 for (var i = 0; i < blockSize; i++) { | 39 for (var i = 0; i < blockSize; i++) { |
37 padding[i] = 0x36 ^ _key[i]; | 40 padding[i] = 0x36 ^ _key[i]; |
38 } | 41 } |
39 | 42 |
40 // Inner hash computation. | 43 // Inner hash computation. |
41 _hash = _hash.newInstance(); | 44 _hash = _hash.newInstance(); |
42 var innerHash = _hash.update(padding).update(_message).digest(); | 45 _hash.add(padding); |
| 46 _hash.add(_message); |
| 47 var innerHash = _hash.close(); |
43 | 48 |
44 // Compute outer padding. | 49 // Compute outer padding. |
45 for (var i = 0; i < blockSize; i++) { | 50 for (var i = 0; i < blockSize; i++) { |
46 padding[i] = 0x5c ^ _key[i]; | 51 padding[i] = 0x5c ^ _key[i]; |
47 } | 52 } |
48 | 53 |
49 // Outer hash computation which is the result. | 54 // Outer hash computation which is the result. |
50 _hash = _hash.newInstance(); | 55 _hash = _hash.newInstance(); |
51 return _hash.update(padding).update(innerHash).digest(); | 56 _hash.add(padding); |
| 57 _hash.add(innerHash); |
| 58 return _hash.close(); |
| 59 } |
| 60 |
| 61 List<int> close() { |
| 62 _isClosed = true; |
| 63 return digest; |
52 } | 64 } |
53 | 65 |
54 bool verify(List<int> digest) { | 66 bool verify(List<int> digest) { |
55 var computedDigest = this.digest(); | 67 var computedDigest = this.digest; |
56 if (digest.length != computedDigest.length) { | 68 if (digest.length != computedDigest.length) { |
57 throw new ArgumentError( | 69 throw new ArgumentError( |
58 'Invalid digest size: ${digest.length} in HMAC.verify. ' | 70 'Invalid digest size: ${digest.length} in HMAC.verify. ' |
59 'Expected: ${_hash.blockSize}.'); | 71 'Expected: ${_hash.blockSize}.'); |
60 } | 72 } |
61 int result = 0; | 73 int result = 0; |
62 for (var i = 0; i < digest.length; i++) { | 74 for (var i = 0; i < digest.length; i++) { |
63 result |= digest[i] ^ computedDigest[i]; | 75 result |= digest[i] ^ computedDigest[i]; |
64 } | 76 } |
65 return result == 0; | 77 return result == 0; |
66 } | 78 } |
67 | 79 |
68 // HMAC internal state. | 80 // HMAC internal state. |
69 Hash _hash; | 81 Hash _hash; |
70 List<int> _key; | 82 List<int> _key; |
71 List<int> _message; | 83 List<int> _message; |
72 } | 84 } |
OLD | NEW |