mss-sdk
Version:
MSS SDK for JavaScript
147 lines (120 loc) • 5.68 kB
text/coffeescript
helpers = require('../helpers')
AWS = helpers.AWS
beforeEach ->
helpers.spyOn(AWS.util, 'userAgent').andReturn('aws-sdk-js/0.1')
buildRequest = ->
ddb = new AWS.DynamoDB({region: 'region', endpoint: 'localhost', apiVersion: '2011-12-05'})
req = ddb.makeRequest('listTables', {ExclusiveStartTableName: 'bår'})
req.build()
req.httpRequest.headers['X-Amz-User-Agent'] = 'aws-sdk-js/0.1'
req.httpRequest
buildSigner = (request) ->
return new AWS.Signers.V4(request || buildRequest(), 'dynamodb')
describe 'AWS.Signers.V4', ->
date = new Date(1935346573456)
datetime = AWS.util.date.iso8601(date).replace(/[:\-]|\.\d{3}/g, '')
creds = null
signature = '0e24aaa0cc86cdc1b73143a147e731cf8c93d450cfcf1d18b2b7473f810b7a1d'
authorization = 'AWS4-HMAC-SHA256 Credential=akid/20310430/region/dynamodb/aws4_request, ' +
'SignedHeaders=host;x-amz-date;x-amz-security-token;x-amz-target;x-amz-user-agent, ' +
'Signature=' + signature
signer = null
beforeEach ->
creds = accessKeyId: 'akid', secretAccessKey: 'secret', sessionToken: 'session'
signer = buildSigner()
signer.addAuthorization(creds, date)
describe 'constructor', ->
it 'can build a signer for a request object', ->
req = buildRequest()
signer = buildSigner(req)
expect(signer.request).to.equal(req)
describe 'addAuthorization', ->
headers = {
'Content-Type': 'application/x-amz-json-1.0',
'Content-Length': 34,
'X-Amz-Target': 'DynamoDB_20111205.ListTables',
'Host': 'localhost',
'X-Amz-Date': datetime,
'x-amz-security-token' : 'session',
'Authorization' : authorization
}
for key, value of headers
func = (k) ->
it 'should add ' + k + ' header', ->
expect(signer.request.headers[k]).to.equal(headers[k])
func(key)
describe 'authorization', ->
it 'should return authorization part for signer', ->
expect(signer.authorization(creds, datetime)).to.equal(authorization)
describe 'signature', ->
it 'should generate proper signature', ->
expect(signer.signature(creds, datetime)).to.equal(signature)
describe 'caching', ->
callCount = null
calls = null
beforeEach ->
helpers.spyOn(AWS.util.crypto, 'hmac')
signer.signature(creds, datetime)
calls = AWS.util.crypto.hmac.calls
callCount = calls.length
it 'caches subsequent requests', ->
signer.signature(creds, datetime)
expect(calls.length).to.equal(callCount + 1)
signer.signature(creds, datetime)
expect(calls.length).to.equal(callCount + 2)
it 'busts cache if region changes', ->
signer.request.region = 'new-region'
signer.signature(creds, datetime)
expect(calls.length).to.equal(callCount + 5)
it 'busts cache if service changes', ->
signer.serviceName = 'newService'
signer.signature(creds, datetime)
expect(calls.length).to.equal(callCount + 5)
it 'busts cache if access key changes', ->
creds.accessKeyId = 'NEWAKID'
signer.signature(creds, datetime)
expect(calls.length).to.equal(callCount + 5)
it 'busts cache if date changes', ->
newDate = new Date(date.getTime() + 1000000000)
newDatetime = AWS.util.date.iso8601(newDate).replace(/[:\-]|\.\d{3}/g, '')
signer.signature(creds, newDatetime)
expect(calls.length).to.equal(callCount + 5)
describe 'stringToSign', ->
it 'should sign correctly generated input string', ->
expect(signer.stringToSign(datetime)).to.equal 'AWS4-HMAC-SHA256\n' +
datetime + '\n' +
'20310430/region/dynamodb/aws4_request\n' +
signer.hexEncodedHash(signer.canonicalString())
describe 'canonicalString', ->
it 'double URI encodes paths for non S3 services', ->
req = new AWS.CognitoSync().listDatasets(IdentityPoolId:'id', IdentityId:'a:b:c').build()
signer = new AWS.Signers.V4(req.httpRequest, 'cognito-identity')
expect(signer.canonicalString().split('\n')[1]).to.equal('/identitypools/id/identities/a%253Ab%253Ac/datasets')
it 'does not double encode path for S3', ->
req = new AWS.S3().getObject(Bucket: 'bucket', Key: 'a:b:c').build()
signer = new AWS.Signers.V4(req.httpRequest, 's3')
expect(signer.canonicalString().split('\n')[1]).to.equal('/a%3Ab%3Ac')
describe 'canonicalHeaders', ->
it 'should return headers', ->
expect(signer.canonicalHeaders()).to.eql [
'host:localhost',
'x-amz-date:' + datetime,
'x-amz-security-token:session',
'x-amz-target:DynamoDB_20111205.ListTables',
'x-amz-user-agent:aws-sdk-js/0.1'
].join('\n')
it 'should ignore Authorization header', ->
signer.request.headers = {'Authorization': 'foo'}
expect(signer.canonicalHeaders()).to.equal('')
it 'should lowercase all header names (not values)', ->
signer.request.headers = {'FOO': 'BAR'}
expect(signer.canonicalHeaders()).to.equal('foo:BAR')
it 'should sort headers by key', ->
signer.request.headers = {abc: 'a', bca: 'b', Qux: 'c', bar: 'd'}
expect(signer.canonicalHeaders()).to.equal('abc:a\nbar:d\nbca:b\nqux:c')
it 'should compact multiple spaces in keys/values to a single space', ->
signer.request.headers = {'Header': 'Value with Multiple \t spaces'}
expect(signer.canonicalHeaders()).to.equal('header:Value with Multiple spaces')
it 'should strip starting and end of line spaces', ->
signer.request.headers = {'Header': ' \t Value \t '}
expect(signer.canonicalHeaders()).to.equal('header:Value')