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
156 lines (154 loc) · 7.83 KB
/
Copy pathCrypt.ts
File metadata and controls
156 lines (154 loc) · 7.83 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
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
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 { Span } from "@opentelemetry/api";
import { Logger } from "./Logger";
export class Crypt {
static encryption_key: string = null; // 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 {
if (NoderedUtil.IsNullUndefinded(user)) throw new Error("user is mandatody")
if (NoderedUtil.IsNullEmpty(password)) throw new Error("password is mandatody")
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 {
if (NoderedUtil.IsNullUndefinded(user)) throw new Error("user is mandatody")
if (NoderedUtil.IsNullEmpty(password)) throw new Error("password is mandatody")
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);
if (NoderedUtil.IsNullEmpty(Crypt.encryption_key)) Crypt.encryption_key = Config.aes_secret.substr(0, 32);
let cipher: crypto.CipherGCM = crypto.createCipheriv('aes-256-gcm', Buffer.from(Crypt.encryption_key), iv);
let encrypted: Buffer = cipher.update((text as any));
encrypted = Buffer.concat([encrypted, cipher.final()]);
const authTag = cipher.getAuthTag()
return iv.toString("hex") + ":" + encrypted.toString("hex") + ":" + authTag.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.shift(), "hex");
let authTag: Buffer = null;
if (textParts.length > 0) authTag = Buffer.from(textParts.shift(), "hex");
let decrypted: Buffer;
if (NoderedUtil.IsNullEmpty(Crypt.encryption_key)) Crypt.encryption_key = Config.aes_secret.substr(0, 32);
if (authTag != null) {
let decipher: crypto.DecipherGCM = crypto.createDecipheriv('aes-256-gcm', Buffer.from(Crypt.encryption_key), iv);
decipher.setAuthTag(authTag);
decrypted = decipher.update(encryptedText);
decrypted = Buffer.concat([decrypted, decipher.final()]);
} else {
let decipher2: crypto.Decipher = crypto.createDecipheriv("aes-256-cbc", Buffer.from(Crypt.encryption_key), iv);
decrypted = decipher2.update(encryptedText);
decrypted = Buffer.concat([decrypted, decipher2.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;
user.customerid = item.customerid;
user.selectedcustomerid = item.selectedcustomerid;
user.dblocked = item.dblocked;
if (NoderedUtil.IsNullEmpty(Crypt.encryption_key)) Crypt.encryption_key = Config.aes_secret.substr(0, 32);
const key = Crypt.encryption_key;
if (NoderedUtil.IsNullEmpty(Config.aes_secret)) throw new Error("Config missing aes_secret");
if (NoderedUtil.IsNullEmpty(key)) throw new Error("Config missing aes_secret");
return jsonwebtoken.sign({ data: user }, key,
{ expiresIn: expiresIn }); // 60 (seconds), "2 days", "10h", "7d"
}
static async verityToken(token: string): Promise<TokenUser> {
if (NoderedUtil.IsNullEmpty(token)) {
throw new Error('jwt must be provided');
}
if (NoderedUtil.IsNullEmpty(Crypt.encryption_key)) Crypt.encryption_key = Config.aes_secret.substr(0, 32);
const o: any = jsonwebtoken.verify(token, Crypt.encryption_key);
let impostor: string = null;
if (!NoderedUtil.IsNullUndefinded(o) && !NoderedUtil.IsNullUndefinded(o.data) && !NoderedUtil.IsNullEmpty(o.data._id)) {
if (!NoderedUtil.IsNullEmpty(o.data.impostor)) {
impostor = o.data.impostor;
}
}
if (!NoderedUtil.IsNullUndefinded(o) && !NoderedUtil.IsNullUndefinded(o.data) && !NoderedUtil.IsNullEmpty(o.data._id) && o.data._id != WellknownIds.root) {
var id = o.data._id;
o.data = await Logger.DBHelper.FindById(o.data._id, token, null);
if (NoderedUtil.IsNullUndefinded(o)) {
var b = true;
}
if (id != o.data._id) {
var b = true;
}
}
if (!NoderedUtil.IsNullEmpty(impostor)) o.data.impostor = impostor;
return TokenUser.assign(o.data);
}
static decryptToken(token: string): any {
if (NoderedUtil.IsNullEmpty(Crypt.encryption_key)) Crypt.encryption_key = Config.aes_secret.substr(0, 32);
return jsonwebtoken.verify(token, Crypt.encryption_key);
}
}