serverless-spy
Version:
CDK-based library for writing elegant integration tests on AWS serverless architecture and an additional web console to monitor events in real time.
95 lines (78 loc) • 2.24 kB
text/typescript
import { BLOCK_SIZE } from "./constants";
import { RawSha256 } from "./RawSha256";
import { Checksum, SourceData } from "@aws-sdk/types";
import { isEmptyData, convertToBuffer } from "@aws-crypto/util";
export class Sha256 implements Checksum {
private readonly secret?: SourceData;
private hash: RawSha256;
private outer?: RawSha256;
private error: any;
constructor(secret?: SourceData) {
this.secret = secret;
this.hash = new RawSha256();
this.reset();
}
update(toHash: SourceData): void {
if (isEmptyData(toHash) || this.error) {
return;
}
try {
this.hash.update(convertToBuffer(toHash));
} catch (e) {
this.error = e;
}
}
/* This synchronous method keeps compatibility
* with the v2 aws-sdk.
*/
digestSync(): Uint8Array {
if (this.error) {
throw this.error;
}
if (this.outer) {
if (!this.outer.finished) {
this.outer.update(this.hash.digest());
}
return this.outer.digest();
}
return this.hash.digest();
}
/* The underlying digest method here is synchronous.
* To keep the same interface with the other hash functions
* the default is to expose this as an async method.
* However, it can sometimes be useful to have a sync method.
*/
async digest(): Promise<Uint8Array> {
return this.digestSync();
}
reset(): void {
this.hash = new RawSha256();
if (this.secret) {
this.outer = new RawSha256();
const inner = bufferFromSecret(this.secret);
const outer = new Uint8Array(BLOCK_SIZE);
outer.set(inner);
for (let i = 0; i < BLOCK_SIZE; i++) {
inner[i] ^= 0x36;
outer[i] ^= 0x5c;
}
this.hash.update(inner);
this.outer.update(outer);
// overwrite the copied key in memory
for (let i = 0; i < inner.byteLength; i++) {
inner[i] = 0;
}
}
}
}
function bufferFromSecret(secret: SourceData): Uint8Array {
let input = convertToBuffer(secret);
if (input.byteLength > BLOCK_SIZE) {
const bufferHash = new RawSha256();
bufferHash.update(input);
input = bufferHash.digest();
}
const buffer = new Uint8Array(BLOCK_SIZE);
buffer.set(input);
return buffer;
}