forked from openiap/opencore
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathCrypt.ts
More file actions
117 lines (115 loc) · 5.45 KB
/
Copy pathCrypt.ts
File metadata and controls
117 lines (115 loc) · 5.45 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
import * as crypto from "crypto";
import * as bcrypt from "bcryptjs";
import * as jsonwebtoken from "jsonwebtoken";
import { Config } from "./Config";
import { NoderedUtil, TokenUser, WellknownIds, Rolemember, User } from "@openiap/openflow-api";
import { Exception } from "handlebars";
import { Span } from "@opentelemetry/api";
import { Logger } from "./Logger";
export class Crypt {
static encryption_key: string = Config.aes_secret.substr(0, 32); // must be 256 bytes (32 characters)
static iv_length: number = 16; // for AES, this is always 16
static bcrypt_salt_rounds: number = 12;
static rootUser(): User {
const result: User = new User();
result._type = "user"; result.name = "root"; result.username = "root"; result._id = WellknownIds.root;
result.roles = []; result.roles.push(new Rolemember("admins", WellknownIds.admins));
return result;
}
static rootToken(): string {
return Crypt.createToken(this.rootUser(), Config.shorttoken_expires_in);
}
public static async SetPassword(user: User, password: string, parent: Span): Promise<void> {
const span: Span = Logger.otel.startSubSpan("Crypt.SetPassword", parent);
try {
user.passwordhash = await Crypt.hash(password);
if (!(this.ValidatePassword(user, password, span))) { throw new Error("Failed validating password after hasing"); }
} catch (error) {
span.recordException(error);
throw error;
} finally {
Logger.otel.endSpan(span);
}
}
public static async ValidatePassword(user: User, password: string, parent: Span): Promise<boolean> {
const span: Span = Logger.otel.startSubSpan("Crypt.ValidatePassword", parent);
try {
return await Crypt.compare(password, user.passwordhash, span);
} catch (error) {
span.recordException(error);
throw error;
} finally {
Logger.otel.endSpan(span);
}
}
static encrypt(text: string): string {
let iv: Buffer = crypto.randomBytes(Crypt.iv_length);
let cipher: crypto.Cipher = crypto.createCipheriv("aes-256-cbc", Buffer.from(Crypt.encryption_key), iv);
let encrypted: Buffer = cipher.update((text as any));
encrypted = Buffer.concat([encrypted, cipher.final()]);
return iv.toString("hex") + ":" + encrypted.toString("hex");
}
static decrypt(text: string): string {
let textParts: string[] = text.split(":");
let iv: Buffer = Buffer.from(textParts.shift(), "hex");
let encryptedText: Buffer = Buffer.from(textParts.join(":"), "hex");
let decipher: crypto.Decipher = crypto.createDecipheriv("aes-256-cbc", Buffer.from(Crypt.encryption_key), iv);
let decrypted: Buffer = decipher.update(encryptedText);
decrypted = Buffer.concat([decrypted, decipher.final()]);
return decrypted.toString();
}
static async hash(password: string): Promise<string> {
return new Promise<string>(async (resolve, reject) => {
try {
bcrypt.hash(password, Crypt.bcrypt_salt_rounds, async (error, hash) => {
if (error) { return reject(error); }
resolve(hash);
});
} catch (error) {
reject(error);
}
});
}
static async compare(password: string, passwordhash: string, parent: Span): Promise<boolean> {
const span: Span = Logger.otel.startSubSpan("Crypt.compare", parent);
return new Promise<boolean>(async (resolve, reject) => {
try {
if (NoderedUtil.IsNullEmpty(password)) { span.recordException("Password cannot be empty"); return reject("Password cannot be empty"); }
if (NoderedUtil.IsNullEmpty(passwordhash)) { span.recordException("Passwordhash cannot be empty"); return reject("Passwordhash cannot be empty"); }
bcrypt.compare(password, passwordhash, async (error, res) => {
if (error) { span.recordException(error); Logger.otel.endSpan(span); return reject(error); }
Logger.otel.endSpan(span);
resolve(res);
});
} catch (error) {
span.recordException(error);
reject(error);
Logger.otel.endSpan(span);
}
});
}
static createToken(item: User | TokenUser, expiresIn: string): string {
const user: TokenUser = new TokenUser();
user._type = (item as User)._type;
user._id = item._id;
user.impostor = (item as TokenUser).impostor;
user.name = item.name;
user.username = item.username;
user.roles = item.roles;
const key = Crypt.encryption_key;
if (NoderedUtil.IsNullEmpty(Config.aes_secret)) throw new Exception("Config missing aes_secret");
if (NoderedUtil.IsNullEmpty(key)) throw new Exception("Config missing aes_secret");
return jsonwebtoken.sign({ data: user }, key,
{ expiresIn: expiresIn }); // 60 (seconds), "2 days", "10h", "7d"
}
static verityToken(token: string): TokenUser {
if (NoderedUtil.IsNullEmpty(token)) {
throw new Error('jwt must be provided');
}
const o: any = jsonwebtoken.verify(token, Crypt.encryption_key);
return TokenUser.assign(o.data);
}
static decryptToken(token: string): any {
return jsonwebtoken.verify(token, Crypt.encryption_key);
}
}