|
10 | 10 | * @param {JsSIP.UA} ua
|
11 | 11 | * @param {JsSIP.OutgoingRequest} request
|
12 | 12 | * @param {JsSIP.IncomingResponse} response
|
13 |
| - * @returns {String} |
14 | 13 | */
|
15 | 14 | JsSIP.DigestAuthentication = function (ua, request, response) {
|
16 |
| - var authenticate, ha1, ha2, param, |
17 |
| - authorization = {}, |
18 |
| - digest = '', |
19 |
| - nc = "00000001", |
20 |
| - cnonce = Math.random().toString(36).substr(2, 12), |
21 |
| - credentials = { |
22 |
| - username: ua.configuration.authorization_user, |
23 |
| - password: ua.configuration.password |
24 |
| - }; |
| 15 | + var authenticate, credentials, realm, qop, nonce, opaque, |
| 16 | + username = ua.configuration.authorization_user, |
| 17 | + password = ua.configuration.password; |
25 | 18 |
|
26 | 19 | if(response.status_code === 401) {
|
27 | 20 | authenticate = response.parseHeader('www-authenticate');
|
28 | 21 | } else {
|
29 | 22 | authenticate = response.parseHeader('proxy-authenticate');
|
30 | 23 | }
|
31 | 24 |
|
32 |
| - response = { |
33 |
| - realm: authenticate.realm.replace(/"/g,''), |
34 |
| - qop: authenticate.qop || null, |
35 |
| - nonce: authenticate.nonce.replace(/"/g,'') |
36 |
| - }; |
| 25 | + realm = authenticate.realm.replace(/"/g,''); |
| 26 | + qop = authenticate.qop || null; |
| 27 | + nonce = authenticate.nonce.replace(/"/g,''); |
| 28 | + opaque = authenticate.opaque; |
| 29 | + |
| 30 | + this.password = password; |
| 31 | + this.method = request.method; |
| 32 | + |
| 33 | + this.username = username; |
| 34 | + this.realm = realm; |
| 35 | + this.nonce = nonce; |
| 36 | + this.uri = request.ruri; |
| 37 | + this.qop = qop; |
| 38 | + this.response = null; |
| 39 | + this.algorithm = "MD5"; |
| 40 | + this.opaque = opaque; |
| 41 | + this.cnonce = null; |
| 42 | + this.nc = 0; |
| 43 | +}; |
| 44 | + |
| 45 | +JsSIP.DigestAuthentication.prototype.authenticate = function(password) { |
| 46 | + var ha1, ha2, nc; |
| 47 | + |
| 48 | + password = password || this.password; |
| 49 | + |
| 50 | + this.cnonce = Math.random().toString(36).substr(2, 12); |
| 51 | + this.nc += 1; |
| 52 | + |
| 53 | + // nc-value = 8LHEX. Max value = 'FFFFFFFF' |
| 54 | + if (this.nc === 4294967296) { |
| 55 | + console.log('Maximum "nc" value has been reached. Reseting "nc"'); |
| 56 | + this.nc = 1; |
| 57 | + } |
37 | 58 |
|
38 | 59 | // HA1 = MD5(A1) = MD5(username:realm:password)
|
39 |
| - ha1 = JsSIP.utils.MD5(credentials.username + ":" + response.realm + ":" + credentials.password); |
40 |
| - |
41 |
| - switch(response.qop) { |
42 |
| - case 'auth-int': |
43 |
| - // HA2 = MD5(A2) = MD5(method:digestURI:MD5(entityBody)) |
44 |
| - ha2 = JsSIP.utils.MD5(request.method + ":" + request.ruri + ":" + JsSIP.utils.MD5(request.body ? request.body : "")); |
45 |
| - break; |
46 |
| - default: |
47 |
| - // HA2 = MD5(A2) = MD5(method:digestURI) |
48 |
| - ha2 = JsSIP.utils.MD5(request.method + ":" + request.ruri); |
| 60 | + ha1 = JsSIP.utils.MD5(this.username + ":" + this.realm + ":" + password); |
| 61 | + |
| 62 | + if (this.qop === 'auth' || this.qop === null) { |
| 63 | + // HA2 = MD5(A2) = MD5(method:digestURI) |
| 64 | + ha2 = JsSIP.utils.MD5(this.method + ":" + this.uri); |
| 65 | + |
| 66 | + } else if (this.qop === 'auth-int') { |
| 67 | + // HA2 = MD5(A2) = MD5(method:digestURI:MD5(entityBody)) |
| 68 | + ha2 = JsSIP.utils.MD5(this.method + ":" + this.uri + ":" + JsSIP.utils.MD5(this.body ? this.body : "")); |
49 | 69 | }
|
50 | 70 |
|
51 |
| - if(response.qop) { |
| 71 | + if(this.qop === 'auth' || this.qop === 'auth-int') { |
52 | 72 | // response = MD5(HA1:nonce:nonceCount:credentialsNonce:qop:HA2)
|
53 |
| - response = JsSIP.utils.MD5(ha1 + ":" + response.nonce + ":" + nc + ":" + cnonce + ":" + response.qop + ":" + ha2); |
| 73 | + this.response = JsSIP.utils.MD5(ha1 + ":" + this.nonce + ":" + this.decimalToHex(this.nc) + ":" + this.cnonce + ":" + this.qop + ":" + ha2); |
54 | 74 | } else {
|
55 | 75 | // response = MD5(HA1:nonce:HA2)
|
56 |
| - response = JsSIP.utils.MD5(ha1 + ":" + response.nonce + ":" + ha2); |
| 76 | + this.response = JsSIP.utils.MD5(ha1 + ":" + this.nonce + ":" + ha2); |
57 | 77 | }
|
58 | 78 |
|
59 |
| - // Fill the Authorization object |
60 |
| - authorization.username = '"' + credentials.username + '"'; |
61 |
| - authorization.realm = authenticate.realm; |
62 |
| - authorization.nonce = authenticate.nonce; |
63 |
| - authorization.uri = '"' + request.ruri + '"'; |
64 |
| - authorization.qop = authenticate.qop || null; |
65 |
| - authorization.response = '"' + response + '"'; |
66 |
| - authorization.algorithm = "MD5"; |
67 |
| - authorization.opaque = authenticate.opaque || null; |
68 |
| - authorization.cnonce = authenticate.qop ? '"' + cnonce + '"' : null; |
69 |
| - authorization.nc = authenticate.qop ? nc : null; |
70 |
| - |
71 |
| - for(param in authorization) { |
72 |
| - if(authorization[param] !== null) { |
73 |
| - digest += ',' + param + '=' + authorization[param]; |
74 |
| - } |
75 |
| - } |
| 79 | + return this.toString(); |
| 80 | +}; |
| 81 | + |
| 82 | +JsSIP.DigestAuthentication.prototype.toString = function() { |
| 83 | + var authorization = 'Digest '; |
| 84 | + |
| 85 | + authorization += 'username="' + this.username + '", '; |
| 86 | + authorization += 'realm="' + this.realm + '", '; |
| 87 | + authorization += 'nonce="' + this.nonce + '", '; |
| 88 | + authorization += 'uri="' + this.uri + '", '; |
| 89 | + authorization += this.qop ? 'qop=' + this.qop + ', ' : ''; |
| 90 | + authorization += 'response="' + this.response + '", '; |
| 91 | + authorization += 'algorithm=MD5, '; |
| 92 | + authorization += this.opaque ? 'opaque="' + this.opaque + '", ': ''; |
| 93 | + authorization += this.qop ? 'cnonce="' + this.cnonce + '", ' : ''; |
| 94 | + authorization += this.qop ? 'nc=' + this.decimalToHex(this.nc) : ''; |
| 95 | + |
| 96 | + return authorization; |
| 97 | +}; |
| 98 | + |
76 | 99 |
|
77 |
| - return 'Digest ' + digest.substr(1); |
| 100 | +JsSIP.DigestAuthentication.prototype.decimalToHex = function(decimal) { |
| 101 | + var hex = Number(decimal).toString(16); |
| 102 | + return '00000000'.substr(0, 8-hex.length) + hex; |
78 | 103 | };
|
0 commit comments