@cubist-labs/cubesigner-sdk 0.3.26 → 0.3.28
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cjs/package.json +1 -1
- package/dist/cjs/src/client.d.ts +4 -4
- package/dist/cjs/src/index.js +7 -4
- package/dist/cjs/src/key.d.ts +35 -2
- package/dist/cjs/src/key.js +79 -5
- package/dist/cjs/src/org_event_processor.d.ts +57 -0
- package/dist/cjs/src/org_event_processor.js +137 -0
- package/dist/cjs/src/schema.d.ts +41 -24
- package/dist/cjs/src/schema.js +1 -1
- package/dist/cjs/src/util.d.ts +2 -0
- package/dist/cjs/src/util.js +1 -1
- package/dist/esm/package.json +1 -1
- package/dist/esm/src/client.d.ts +4 -4
- package/dist/esm/src/index.js +4 -4
- package/dist/esm/src/key.d.ts +35 -2
- package/dist/esm/src/key.js +79 -5
- package/dist/esm/src/org_event_processor.d.ts +57 -0
- package/dist/esm/src/org_event_processor.js +133 -0
- package/dist/esm/src/schema.d.ts +41 -24
- package/dist/esm/src/schema.js +1 -1
- package/dist/esm/src/util.d.ts +2 -0
- package/dist/esm/src/util.js +1 -1
- package/dist/package.json +36 -0
- package/dist/spec/env/beta.json +9 -0
- package/dist/spec/env/gamma.json +9 -0
- package/dist/spec/env/prod.json +9 -0
- package/dist/src/api.d.ts +634 -0
- package/dist/src/api.js +1309 -0
- package/dist/src/client.d.ts +575 -0
- package/dist/src/client.js +381 -0
- package/dist/src/env.d.ts +15 -0
- package/dist/src/env.js +35 -0
- package/dist/src/error.d.ts +29 -0
- package/dist/src/error.js +36 -0
- package/dist/src/events.d.ts +84 -0
- package/dist/src/events.js +195 -0
- package/dist/src/index.d.ts +207 -0
- package/dist/src/index.js +308 -0
- package/dist/src/key.d.ts +152 -0
- package/dist/src/key.js +242 -0
- package/dist/src/mfa.d.ts +94 -0
- package/dist/src/mfa.js +169 -0
- package/dist/src/org.d.ts +99 -0
- package/dist/src/org.js +95 -0
- package/dist/src/paginator.d.ts +76 -0
- package/dist/src/paginator.js +99 -0
- package/dist/src/response.d.ts +101 -0
- package/dist/src/response.js +164 -0
- package/dist/src/role.d.ts +283 -0
- package/dist/src/role.js +253 -0
- package/dist/src/schema.d.ts +6209 -0
- package/dist/src/schema.js +7 -0
- package/dist/src/schema_types.d.ts +113 -0
- package/dist/src/schema_types.js +3 -0
- package/dist/src/session/session_storage.d.ts +47 -0
- package/dist/src/session/session_storage.js +76 -0
- package/dist/src/session/signer_session_manager.d.ts +125 -0
- package/dist/src/session/signer_session_manager.js +239 -0
- package/dist/src/signer_session.d.ts +41 -0
- package/dist/src/signer_session.js +77 -0
- package/dist/src/user_export.d.ts +52 -0
- package/dist/src/user_export.js +129 -0
- package/dist/src/util.d.ts +61 -0
- package/dist/src/util.js +97 -0
- package/package.json +1 -1
- package/src/index.ts +3 -3
- package/src/key.ts +83 -4
- package/src/schema.ts +42 -25
- package/src/util.ts +3 -0
|
@@ -0,0 +1,239 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
|
|
3
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
|
|
4
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
|
|
5
|
+
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
|
|
6
|
+
};
|
|
7
|
+
var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
|
|
8
|
+
if (kind === "m") throw new TypeError("Private method is not writable");
|
|
9
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
|
|
10
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
|
|
11
|
+
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
|
|
12
|
+
};
|
|
13
|
+
var _SignerSessionManager_instances, _a, _SignerSessionManager_eventEmitter, _SignerSessionManager_refreshing, _SignerSessionManager_client, _SignerSessionManager_createClient, _SignerSessionManager_hasTimestampExpired;
|
|
14
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
15
|
+
exports.SignerSessionManager = void 0;
|
|
16
|
+
const events_1 = require("../events");
|
|
17
|
+
const api_1 = require("../api");
|
|
18
|
+
const session_storage_1 = require("./session_storage");
|
|
19
|
+
const util_1 = require("../util");
|
|
20
|
+
const events_2 = require("../events");
|
|
21
|
+
const error_1 = require("../error");
|
|
22
|
+
const DEFAULT_EXPIRATION_BUFFER_SECS = 30;
|
|
23
|
+
/**
|
|
24
|
+
* Constructs {@link Date} from a number representing seconds since unix epoch.
|
|
25
|
+
* @param {number} secs Seconds since unix epoch.
|
|
26
|
+
* @return {Date} The equivalent date.
|
|
27
|
+
*/
|
|
28
|
+
function secondsSinceEpochToDate(secs) {
|
|
29
|
+
return new Date(secs * 1000);
|
|
30
|
+
}
|
|
31
|
+
/** Generic session manager interface. */
|
|
32
|
+
class SignerSessionManager {
|
|
33
|
+
/**
|
|
34
|
+
* @return {string} The current auth token.
|
|
35
|
+
* @internal
|
|
36
|
+
*/
|
|
37
|
+
async token() {
|
|
38
|
+
const session = await this.storage.retrieve();
|
|
39
|
+
return session.token;
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Refreshes the current session if needed, then returns a client using the current session.
|
|
43
|
+
*
|
|
44
|
+
* May **UPDATE/MUTATE** self.
|
|
45
|
+
*
|
|
46
|
+
* @param {operations} operation The operation that this client will be
|
|
47
|
+
* used for. This parameter is used exclusively for more accurate error
|
|
48
|
+
* reporting and does not affect functionality.
|
|
49
|
+
* @return {Client} The client with the current session
|
|
50
|
+
*/
|
|
51
|
+
async client(operation) {
|
|
52
|
+
await this.refreshIfNeeded();
|
|
53
|
+
// trigger "session expired" if the session as a whole has expired
|
|
54
|
+
// or if (for whatever reason) the token is still stale
|
|
55
|
+
if (__classPrivateFieldGet(_a, _a, "m", _SignerSessionManager_hasTimestampExpired).call(_a, __classPrivateFieldGet(this, _SignerSessionManager_client, "f").token_exp) || this.hasExpired()) {
|
|
56
|
+
await __classPrivateFieldGet(this, _SignerSessionManager_eventEmitter, "f").emitSessionExpired();
|
|
57
|
+
throw new error_1.SessionExpiredError(operation);
|
|
58
|
+
}
|
|
59
|
+
return __classPrivateFieldGet(this, _SignerSessionManager_client, "f").client;
|
|
60
|
+
}
|
|
61
|
+
/** Revokes the session. */
|
|
62
|
+
async revoke() {
|
|
63
|
+
const client = new api_1.OpClient("revokeCurrentSession", await this.client(), __classPrivateFieldGet(this, _SignerSessionManager_eventEmitter, "f"));
|
|
64
|
+
await client.del("/v0/org/{org_id}/session/self", {
|
|
65
|
+
params: { path: { org_id: this.orgId } },
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Refreshes the session and **UPDATES/MUTATES** self.
|
|
70
|
+
*/
|
|
71
|
+
async refresh() {
|
|
72
|
+
if (this.hasExpired()) {
|
|
73
|
+
await __classPrivateFieldGet(this, _SignerSessionManager_eventEmitter, "f").emitSessionExpired();
|
|
74
|
+
throw new error_1.SessionExpiredError("signerSessionRefresh");
|
|
75
|
+
}
|
|
76
|
+
const currSession = await this.storage.retrieve();
|
|
77
|
+
const client = new api_1.OpClient("signerSessionRefresh", __classPrivateFieldGet(this, _SignerSessionManager_client, "f").client, __classPrivateFieldGet(this, _SignerSessionManager_eventEmitter, "f"));
|
|
78
|
+
const csi = currSession.session_info;
|
|
79
|
+
const data = await client.patch("/v1/org/{org_id}/token/refresh", {
|
|
80
|
+
params: { path: { org_id: this.orgId } },
|
|
81
|
+
body: {
|
|
82
|
+
epoch_num: csi.epoch,
|
|
83
|
+
epoch_token: csi.epoch_token,
|
|
84
|
+
other_token: csi.refresh_token,
|
|
85
|
+
},
|
|
86
|
+
});
|
|
87
|
+
const newSession = {
|
|
88
|
+
...currSession,
|
|
89
|
+
session_info: data.session_info,
|
|
90
|
+
token: data.token,
|
|
91
|
+
};
|
|
92
|
+
await this.storage.save(newSession);
|
|
93
|
+
__classPrivateFieldSet(this, _SignerSessionManager_client, {
|
|
94
|
+
client: __classPrivateFieldGet(this, _SignerSessionManager_instances, "m", _SignerSessionManager_createClient).call(this, newSession.token),
|
|
95
|
+
token_exp: secondsSinceEpochToDate(newSession.session_info.auth_token_exp),
|
|
96
|
+
session_exp: newSession.session_exp
|
|
97
|
+
? secondsSinceEpochToDate(newSession.session_exp)
|
|
98
|
+
: undefined,
|
|
99
|
+
}, "f");
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Returns whether it's time to refresh this token.
|
|
103
|
+
* @return {boolean} Whether it's time to refresh this token.
|
|
104
|
+
* @internal
|
|
105
|
+
*/
|
|
106
|
+
async isStale() {
|
|
107
|
+
return __classPrivateFieldGet(_a, _a, "m", _SignerSessionManager_hasTimestampExpired).call(_a, __classPrivateFieldGet(this, _SignerSessionManager_client, "f").token_exp, DEFAULT_EXPIRATION_BUFFER_SECS);
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Return whether this session has expired and cannot be refreshed anymore.
|
|
111
|
+
* @return {boolean} Whether this session has expired.
|
|
112
|
+
* @internal
|
|
113
|
+
*/
|
|
114
|
+
hasExpired() {
|
|
115
|
+
return ((__classPrivateFieldGet(this, _SignerSessionManager_client, "f").session_exp || false) &&
|
|
116
|
+
__classPrivateFieldGet(_a, _a, "m", _SignerSessionManager_hasTimestampExpired).call(_a, __classPrivateFieldGet(this, _SignerSessionManager_client, "f").session_exp));
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Refreshes the session if it is about to expire.
|
|
120
|
+
* @return {boolean} Whether the session token was refreshed.
|
|
121
|
+
* @internal
|
|
122
|
+
*/
|
|
123
|
+
async refreshIfNeeded() {
|
|
124
|
+
if (await this.isStale()) {
|
|
125
|
+
if (__classPrivateFieldGet(this, _SignerSessionManager_refreshing, "f")) {
|
|
126
|
+
// wait until done refreshing
|
|
127
|
+
while (__classPrivateFieldGet(this, _SignerSessionManager_refreshing, "f")) {
|
|
128
|
+
await (0, util_1.delay)(100);
|
|
129
|
+
}
|
|
130
|
+
return false;
|
|
131
|
+
}
|
|
132
|
+
else {
|
|
133
|
+
// refresh
|
|
134
|
+
__classPrivateFieldSet(this, _SignerSessionManager_refreshing, true, "f");
|
|
135
|
+
try {
|
|
136
|
+
await this.refresh();
|
|
137
|
+
return true;
|
|
138
|
+
}
|
|
139
|
+
finally {
|
|
140
|
+
__classPrivateFieldSet(this, _SignerSessionManager_refreshing, false, "f");
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
return false;
|
|
145
|
+
}
|
|
146
|
+
/**
|
|
147
|
+
* Automatically refreshes the session in the background (if needed) every
|
|
148
|
+
* minute. This is a simple wrapper around `setInterval`.
|
|
149
|
+
* @return {number} The interval ID of the refresh timer.
|
|
150
|
+
*/
|
|
151
|
+
autoRefresh() {
|
|
152
|
+
return setInterval(async () => {
|
|
153
|
+
await this.refreshIfNeeded();
|
|
154
|
+
}, 60 * 1000);
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* Clears the auto refresh timer.
|
|
158
|
+
* @param {number} timer The timer ID to clear.
|
|
159
|
+
*/
|
|
160
|
+
clearAutoRefresh(timer) {
|
|
161
|
+
clearInterval(timer);
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* @param {EnvInterface} env The CubeSigner environment
|
|
165
|
+
* @param {string} orgId The organization ID
|
|
166
|
+
* @param {NewSessionResponse} session The session information.
|
|
167
|
+
* @param {SignerSessionStorage} storage The storage to use for saving the session.
|
|
168
|
+
* @return {Promise<SignerSessionManager>} New signer session manager.
|
|
169
|
+
*/
|
|
170
|
+
static async createFromSessionInfo(env, orgId, session, storage) {
|
|
171
|
+
const sessionData = {
|
|
172
|
+
env: {
|
|
173
|
+
["Dev-CubeSignerStack"]: env,
|
|
174
|
+
},
|
|
175
|
+
org_id: orgId,
|
|
176
|
+
token: session.token,
|
|
177
|
+
purpose: "sign via oidc",
|
|
178
|
+
session_info: session.session_info,
|
|
179
|
+
session_exp: session.expiration,
|
|
180
|
+
};
|
|
181
|
+
storage ??= new session_storage_1.MemorySessionStorage();
|
|
182
|
+
await storage.save(sessionData);
|
|
183
|
+
return await _a.loadFromStorage(storage);
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* @param {SignerSessionData} sessionData The session information.
|
|
187
|
+
* @param {SignerSessionStorage} storage The storage to use for saving the session.
|
|
188
|
+
* @return {Promise<SignerSessionManager>} New signer session manager.
|
|
189
|
+
*/
|
|
190
|
+
static async createFromSessionData(sessionData, storage) {
|
|
191
|
+
storage ??= new session_storage_1.MemorySessionStorage();
|
|
192
|
+
await storage.save(sessionData);
|
|
193
|
+
return await _a.loadFromStorage(storage);
|
|
194
|
+
}
|
|
195
|
+
/**
|
|
196
|
+
* Uses an existing session to create a new signer session manager.
|
|
197
|
+
*
|
|
198
|
+
* @param {SignerSessionStorage} storage The session storage to use
|
|
199
|
+
* @return {Promise<SingerSession>} New signer session manager
|
|
200
|
+
*/
|
|
201
|
+
static async loadFromStorage(storage) {
|
|
202
|
+
const session = await storage.retrieve();
|
|
203
|
+
return new _a(session, storage);
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Constructor.
|
|
207
|
+
* @param {SignerSessionData} sessionData Session data
|
|
208
|
+
* @param {SignerSessionStorage} storage The session storage to use.
|
|
209
|
+
*/
|
|
210
|
+
constructor(sessionData, storage) {
|
|
211
|
+
_SignerSessionManager_instances.add(this);
|
|
212
|
+
this.events = new events_1.Events();
|
|
213
|
+
_SignerSessionManager_eventEmitter.set(this, void 0);
|
|
214
|
+
_SignerSessionManager_refreshing.set(this, false);
|
|
215
|
+
_SignerSessionManager_client.set(this, void 0);
|
|
216
|
+
this.env = sessionData.env["Dev-CubeSignerStack"];
|
|
217
|
+
this.orgId = sessionData.org_id;
|
|
218
|
+
this.storage = storage;
|
|
219
|
+
__classPrivateFieldSet(this, _SignerSessionManager_eventEmitter, new events_2.EventEmitter([this.events]), "f");
|
|
220
|
+
__classPrivateFieldSet(this, _SignerSessionManager_client, {
|
|
221
|
+
client: __classPrivateFieldGet(this, _SignerSessionManager_instances, "m", _SignerSessionManager_createClient).call(this, sessionData.token),
|
|
222
|
+
token_exp: secondsSinceEpochToDate(sessionData.session_info.auth_token_exp),
|
|
223
|
+
session_exp: sessionData.session_exp
|
|
224
|
+
? secondsSinceEpochToDate(sessionData.session_exp)
|
|
225
|
+
: undefined,
|
|
226
|
+
}, "f");
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
exports.SignerSessionManager = SignerSessionManager;
|
|
230
|
+
_a = SignerSessionManager, _SignerSessionManager_eventEmitter = new WeakMap(), _SignerSessionManager_refreshing = new WeakMap(), _SignerSessionManager_client = new WeakMap(), _SignerSessionManager_instances = new WeakSet(), _SignerSessionManager_createClient = function _SignerSessionManager_createClient(token) {
|
|
231
|
+
return (0, api_1.createHttpClient)(this.env.SignerApiRoot, token);
|
|
232
|
+
}, _SignerSessionManager_hasTimestampExpired = function _SignerSessionManager_hasTimestampExpired(exp, bufferSeconds) {
|
|
233
|
+
bufferSeconds ??= 0;
|
|
234
|
+
const expMsSinceEpoch = exp.getTime();
|
|
235
|
+
const nowMsSinceEpoch = new Date().getTime();
|
|
236
|
+
const bufferMs = bufferSeconds * 1000;
|
|
237
|
+
return expMsSinceEpoch < nowMsSinceEpoch + bufferMs;
|
|
238
|
+
};
|
|
239
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signer_session_manager.js","sourceRoot":"","sources":["../../../src/session/signer_session_manager.ts"],"names":[],"mappings":"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","sourcesContent":["import { Events } from \"../events\";\nimport { EnvInterface } from \"../env\";\nimport { Client, createHttpClient, OpClient } from \"../api\";\nimport { MemorySessionStorage, SessionStorage } from \"./session_storage\";\nimport { delay } from \"../util\";\nimport {\n  ClientSessionInfo,\n  NewSessionResponse,\n  RefreshSignerSessionRequest,\n} from \"../schema_types\";\nimport { EventEmitter } from \"../events\";\nimport { SessionExpiredError } from \"../error\";\nimport { operations } from \"../schema\";\n\nconst DEFAULT_EXPIRATION_BUFFER_SECS = 30;\n\n/** JSON representation of our \"signer session\" file format */\nexport interface SignerSessionData {\n  /** The organization ID */\n  org_id: string;\n  /** The role ID */\n  role_id?: string;\n  /** The purpose of the session token */\n  purpose?: string;\n  /** The token to include in Authorization header */\n  token: string;\n  /** Session info */\n  session_info: ClientSessionInfo;\n  /** Session expiration (in seconds since UNIX epoch) beyond which it cannot be refreshed */\n  session_exp: number | undefined; // may be missing in legacy session files\n  /** The environment */\n  env: {\n    [\"Dev-CubeSignerStack\"]: EnvInterface;\n  };\n}\n\n/**\n * Constructs {@link Date} from a number representing seconds since unix epoch.\n * @param {number} secs Seconds since unix epoch.\n * @return {Date} The equivalent date.\n */\nfunction secondsSinceEpochToDate(secs: number): Date {\n  return new Date(secs * 1000);\n}\n\n/** Type of storage required for signer sessions */\nexport type SignerSessionStorage = SessionStorage<SignerSessionData>;\n\nexport interface SignerSessionLifetime {\n  /** Session lifetime (in seconds). Defaults to one week (604800). */\n  session?: number;\n  /** Auth token lifetime (in seconds). Defaults to five minutes (300). */\n  auth: number;\n  /** Refresh token lifetime (in seconds). Defaults to one day (86400). */\n  refresh?: number;\n  /** Grace lifetime (in seconds). Defaults to 30 seconds (30). */\n  grace?: number;\n}\n\n/** Generic session manager interface. */\nexport class SignerSessionManager {\n  readonly env: EnvInterface;\n  readonly orgId: string;\n  readonly storage: SignerSessionStorage;\n  readonly events = new Events();\n  readonly #eventEmitter: EventEmitter;\n  #refreshing: boolean = false;\n  #client: { client: Client; token_exp: Date; session_exp?: Date };\n\n  /**\n   * @return {string} The current auth token.\n   * @internal\n   */\n  async token(): Promise<string> {\n    const session = await this.storage.retrieve();\n    return session.token;\n  }\n\n  /**\n   * Refreshes the current session if needed, then returns a client using the current session.\n   *\n   * May **UPDATE/MUTATE** self.\n   *\n   * @param {operations} operation The operation that this client will be\n   *   used for. This parameter is used exclusively for more accurate error\n   *   reporting and does not affect functionality.\n   * @return {Client} The client with the current session\n   */\n  async client(operation?: keyof operations): Promise<Client> {\n    await this.refreshIfNeeded();\n\n    // trigger \"session expired\" if the session as a whole has expired\n    // or if (for whatever reason) the token is still stale\n    if (SignerSessionManager.#hasTimestampExpired(this.#client.token_exp) || this.hasExpired()) {\n      await this.#eventEmitter.emitSessionExpired();\n      throw new SessionExpiredError(operation);\n    }\n\n    return this.#client.client;\n  }\n\n  /** Revokes the session. */\n  async revoke(): Promise<void> {\n    const client = new OpClient(\"revokeCurrentSession\", await this.client(), this.#eventEmitter);\n    await client.del(\"/v0/org/{org_id}/session/self\", {\n      params: { path: { org_id: this.orgId } },\n    });\n  }\n\n  /**\n   * Refreshes the session and **UPDATES/MUTATES** self.\n   */\n  async refresh(): Promise<void> {\n    if (this.hasExpired()) {\n      await this.#eventEmitter.emitSessionExpired();\n      throw new SessionExpiredError(\"signerSessionRefresh\");\n    }\n\n    const currSession = await this.storage.retrieve();\n\n    const client = new OpClient(\"signerSessionRefresh\", this.#client.client, this.#eventEmitter);\n    const csi = currSession.session_info;\n    const data = await client.patch(\"/v1/org/{org_id}/token/refresh\", {\n      params: { path: { org_id: this.orgId } },\n      body: <RefreshSignerSessionRequest>{\n        epoch_num: csi.epoch,\n        epoch_token: csi.epoch_token,\n        other_token: csi.refresh_token,\n      },\n    });\n    const newSession = <SignerSessionData>{\n      ...currSession,\n      session_info: data.session_info,\n      token: data.token,\n    };\n\n    await this.storage.save(newSession);\n    this.#client = {\n      client: this.#createClient(newSession.token),\n      token_exp: secondsSinceEpochToDate(newSession.session_info.auth_token_exp),\n      session_exp: newSession.session_exp\n        ? secondsSinceEpochToDate(newSession.session_exp)\n        : undefined,\n    };\n  }\n\n  /**\n   * Returns whether it's time to refresh this token.\n   * @return {boolean} Whether it's time to refresh this token.\n   * @internal\n   */\n  async isStale(): Promise<boolean> {\n    return SignerSessionManager.#hasTimestampExpired(\n      this.#client.token_exp,\n      DEFAULT_EXPIRATION_BUFFER_SECS,\n    );\n  }\n\n  /**\n   * Return whether this session has expired and cannot be refreshed anymore.\n   * @return {boolean} Whether this session has expired.\n   * @internal\n   */\n  hasExpired(): boolean {\n    return (\n      (this.#client.session_exp || false) &&\n      SignerSessionManager.#hasTimestampExpired(this.#client.session_exp)\n    );\n  }\n\n  /**\n   * Refreshes the session if it is about to expire.\n   * @return {boolean} Whether the session token was refreshed.\n   * @internal\n   */\n  async refreshIfNeeded(): Promise<boolean> {\n    if (await this.isStale()) {\n      if (this.#refreshing) {\n        // wait until done refreshing\n        while (this.#refreshing) {\n          await delay(100);\n        }\n        return false;\n      } else {\n        // refresh\n        this.#refreshing = true;\n        try {\n          await this.refresh();\n          return true;\n        } finally {\n          this.#refreshing = false;\n        }\n      }\n    }\n\n    return false;\n  }\n\n  /**\n   * Automatically refreshes the session in the background (if needed) every\n   * minute. This is a simple wrapper around `setInterval`.\n   * @return {number} The interval ID of the refresh timer.\n   */\n  autoRefresh(): RefreshId {\n    return setInterval(async () => {\n      await this.refreshIfNeeded();\n    }, 60 * 1000);\n  }\n\n  /**\n   * Clears the auto refresh timer.\n   * @param {number} timer The timer ID to clear.\n   */\n  clearAutoRefresh(timer: RefreshId): void {\n    clearInterval(timer);\n  }\n\n  /**\n   * @param {EnvInterface} env The CubeSigner environment\n   * @param {string} orgId The organization ID\n   * @param {NewSessionResponse} session The session information.\n   * @param {SignerSessionStorage} storage The storage to use for saving the session.\n   * @return {Promise<SignerSessionManager>} New signer session manager.\n   */\n  static async createFromSessionInfo(\n    env: EnvInterface,\n    orgId: string,\n    session: NewSessionResponse,\n    storage?: SignerSessionStorage,\n  ): Promise<SignerSessionManager> {\n    const sessionData = {\n      env: {\n        [\"Dev-CubeSignerStack\"]: env,\n      },\n      org_id: orgId,\n      token: session.token,\n      purpose: \"sign via oidc\",\n      session_info: session.session_info,\n      session_exp: session.expiration!,\n    };\n    storage ??= new MemorySessionStorage();\n    await storage.save(sessionData);\n    return await SignerSessionManager.loadFromStorage(storage);\n  }\n\n  /**\n   * @param {SignerSessionData} sessionData The session information.\n   * @param {SignerSessionStorage} storage The storage to use for saving the session.\n   * @return {Promise<SignerSessionManager>} New signer session manager.\n   */\n  static async createFromSessionData(\n    sessionData: SignerSessionData,\n    storage?: SignerSessionStorage,\n  ): Promise<SignerSessionManager> {\n    storage ??= new MemorySessionStorage();\n    await storage.save(sessionData);\n    return await SignerSessionManager.loadFromStorage(storage);\n  }\n\n  /**\n   * Uses an existing session to create a new signer session manager.\n   *\n   * @param {SignerSessionStorage} storage The session storage to use\n   * @return {Promise<SingerSession>} New signer session manager\n   */\n  static async loadFromStorage(storage: SignerSessionStorage): Promise<SignerSessionManager> {\n    const session = await storage.retrieve();\n    return new SignerSessionManager(session, storage);\n  }\n\n  /**\n   * Constructor.\n   * @param {SignerSessionData} sessionData Session data\n   * @param {SignerSessionStorage} storage The session storage to use.\n   */\n  constructor(sessionData: SignerSessionData, storage: SignerSessionStorage) {\n    this.env = sessionData.env[\"Dev-CubeSignerStack\"];\n    this.orgId = sessionData.org_id;\n    this.storage = storage;\n    this.#eventEmitter = new EventEmitter([this.events]);\n    this.#client = {\n      client: this.#createClient(sessionData.token),\n      token_exp: secondsSinceEpochToDate(sessionData.session_info.auth_token_exp),\n      session_exp: sessionData.session_exp\n        ? secondsSinceEpochToDate(sessionData.session_exp)\n        : undefined,\n    };\n  }\n\n  /**\n   * Creates a new REST client with a given token\n   * @param {string} token The authorization token to use for the client\n   * @return {Client} The new REST client\n   */\n  #createClient(token: string): Client {\n    return createHttpClient(this.env.SignerApiRoot, token);\n  }\n\n  /**\n   * Check if a timestamp is within {@link bufferSeconds} seconds from expiration.\n   * @param {Date} exp The timestamp to check\n   * @param {number} bufferSeconds Time buffer in seconds (defaults to 0s)\n   * @return {boolean} True if the timestamp has expired\n   */\n  static #hasTimestampExpired(exp: Date, bufferSeconds?: number): boolean {\n    bufferSeconds ??= 0;\n    const expMsSinceEpoch = exp.getTime();\n    const nowMsSinceEpoch = new Date().getTime();\n    const bufferMs = bufferSeconds * 1000;\n    return expMsSinceEpoch < nowMsSinceEpoch + bufferMs;\n  }\n}\n\n/** Type of the refresh timer ID. */\nexport type RefreshId = ReturnType<typeof setInterval>;\n"]}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { CubeSignerClient } from "./client";
|
|
2
|
+
import { KeyInfo } from "./key";
|
|
3
|
+
import { SignerSessionManager, SignerSessionStorage } from "./session/signer_session_manager";
|
|
4
|
+
/** Signer session info. Can only be used to revoke a token, but not for authentication. */
|
|
5
|
+
export declare class SignerSessionInfo {
|
|
6
|
+
#private;
|
|
7
|
+
readonly purpose: string;
|
|
8
|
+
/** Revoke this session */
|
|
9
|
+
revoke(): Promise<void>;
|
|
10
|
+
/**
|
|
11
|
+
* Internal constructor.
|
|
12
|
+
* @param {CubeSignerClient} cs CubeSigner instance to use when calling `revoke`
|
|
13
|
+
* @param {string} sessionId The ID of the session; can be used for revocation but not for auth
|
|
14
|
+
* @param {string} purpose Session purpose
|
|
15
|
+
* @internal
|
|
16
|
+
*/
|
|
17
|
+
constructor(cs: CubeSignerClient, sessionId: string, purpose: string);
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Signer session.
|
|
21
|
+
* Extends {@link CubeSignerClient} and provides a few convenience methods on top.
|
|
22
|
+
*/
|
|
23
|
+
export declare class SignerSession extends CubeSignerClient {
|
|
24
|
+
/**
|
|
25
|
+
* Loads an existing signer session from storage.
|
|
26
|
+
* @param {SignerSessionStorage} storage The session storage to use
|
|
27
|
+
* @return {Promise<SingerSession>} New signer session
|
|
28
|
+
*/
|
|
29
|
+
static loadSignerSession(storage: SignerSessionStorage): Promise<SignerSession>;
|
|
30
|
+
/**
|
|
31
|
+
* Constructor.
|
|
32
|
+
* @param {SignerSessionManager} sessionMgr The session manager to use
|
|
33
|
+
* @internal
|
|
34
|
+
*/
|
|
35
|
+
constructor(sessionMgr: SignerSessionManager);
|
|
36
|
+
/**
|
|
37
|
+
* Returns the list of keys that this token grants access to.
|
|
38
|
+
* @return {KeyInfo[]} The list of keys.
|
|
39
|
+
*/
|
|
40
|
+
keys(): Promise<KeyInfo[]>;
|
|
41
|
+
}
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
|
|
3
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
|
|
4
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
|
|
5
|
+
return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
|
|
6
|
+
};
|
|
7
|
+
var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
|
|
8
|
+
if (kind === "m") throw new TypeError("Private method is not writable");
|
|
9
|
+
if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
|
|
10
|
+
if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
|
|
11
|
+
return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
|
|
12
|
+
};
|
|
13
|
+
var _SignerSessionInfo_csc, _SignerSessionInfo_sessionId;
|
|
14
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
15
|
+
exports.SignerSession = exports.SignerSessionInfo = void 0;
|
|
16
|
+
const client_1 = require("./client");
|
|
17
|
+
const key_1 = require("./key");
|
|
18
|
+
const signer_session_manager_1 = require("./session/signer_session_manager");
|
|
19
|
+
/** Signer session info. Can only be used to revoke a token, but not for authentication. */
|
|
20
|
+
class SignerSessionInfo {
|
|
21
|
+
/** Revoke this session */
|
|
22
|
+
async revoke() {
|
|
23
|
+
await __classPrivateFieldGet(this, _SignerSessionInfo_csc, "f").sessionRevoke(__classPrivateFieldGet(this, _SignerSessionInfo_sessionId, "f"));
|
|
24
|
+
}
|
|
25
|
+
// --------------------------------------------------------------------------
|
|
26
|
+
// -- INTERNAL --------------------------------------------------------------
|
|
27
|
+
// --------------------------------------------------------------------------
|
|
28
|
+
/**
|
|
29
|
+
* Internal constructor.
|
|
30
|
+
* @param {CubeSignerClient} cs CubeSigner instance to use when calling `revoke`
|
|
31
|
+
* @param {string} sessionId The ID of the session; can be used for revocation but not for auth
|
|
32
|
+
* @param {string} purpose Session purpose
|
|
33
|
+
* @internal
|
|
34
|
+
*/
|
|
35
|
+
constructor(cs, sessionId, purpose) {
|
|
36
|
+
_SignerSessionInfo_csc.set(this, void 0);
|
|
37
|
+
_SignerSessionInfo_sessionId.set(this, void 0);
|
|
38
|
+
__classPrivateFieldSet(this, _SignerSessionInfo_csc, cs, "f");
|
|
39
|
+
__classPrivateFieldSet(this, _SignerSessionInfo_sessionId, sessionId, "f");
|
|
40
|
+
this.purpose = purpose;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
exports.SignerSessionInfo = SignerSessionInfo;
|
|
44
|
+
_SignerSessionInfo_csc = new WeakMap(), _SignerSessionInfo_sessionId = new WeakMap();
|
|
45
|
+
/**
|
|
46
|
+
* Signer session.
|
|
47
|
+
* Extends {@link CubeSignerClient} and provides a few convenience methods on top.
|
|
48
|
+
*/
|
|
49
|
+
class SignerSession extends client_1.CubeSignerClient {
|
|
50
|
+
/**
|
|
51
|
+
* Loads an existing signer session from storage.
|
|
52
|
+
* @param {SignerSessionStorage} storage The session storage to use
|
|
53
|
+
* @return {Promise<SingerSession>} New signer session
|
|
54
|
+
*/
|
|
55
|
+
static async loadSignerSession(storage) {
|
|
56
|
+
const manager = await signer_session_manager_1.SignerSessionManager.loadFromStorage(storage);
|
|
57
|
+
return new SignerSession(manager);
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Constructor.
|
|
61
|
+
* @param {SignerSessionManager} sessionMgr The session manager to use
|
|
62
|
+
* @internal
|
|
63
|
+
*/
|
|
64
|
+
constructor(sessionMgr) {
|
|
65
|
+
super(sessionMgr);
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Returns the list of keys that this token grants access to.
|
|
69
|
+
* @return {KeyInfo[]} The list of keys.
|
|
70
|
+
*/
|
|
71
|
+
async keys() {
|
|
72
|
+
const keys = await this.sessionKeysList();
|
|
73
|
+
return keys.map((k) => (0, key_1.toKeyInfo)(k));
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
exports.SignerSession = SignerSession;
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2lnbmVyX3Nlc3Npb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc2lnbmVyX3Nlc3Npb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7O0FBQUEscUNBQTRDO0FBQzVDLCtCQUEyQztBQUMzQyw2RUFBOEY7QUFFOUYsMkZBQTJGO0FBQzNGLE1BQWEsaUJBQWlCO0lBSzVCLDBCQUEwQjtJQUMxQixLQUFLLENBQUMsTUFBTTtRQUNWLE1BQU0sdUJBQUEsSUFBSSw4QkFBSyxDQUFDLGFBQWEsQ0FBQyx1QkFBQSxJQUFJLG9DQUFXLENBQUMsQ0FBQztJQUNqRCxDQUFDO0lBRUQsNkVBQTZFO0lBQzdFLDZFQUE2RTtJQUM3RSw2RUFBNkU7SUFFN0U7Ozs7OztPQU1HO0lBQ0gsWUFBWSxFQUFvQixFQUFFLFNBQWlCLEVBQUUsT0FBZTtRQXBCM0QseUNBQXVCO1FBQ3ZCLCtDQUFtQjtRQW9CMUIsdUJBQUEsSUFBSSwwQkFBUSxFQUFFLE1BQUEsQ0FBQztRQUNmLHVCQUFBLElBQUksZ0NBQWMsU0FBUyxNQUFBLENBQUM7UUFDNUIsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUM7SUFDekIsQ0FBQztDQUNGO0FBMUJELDhDQTBCQzs7QUFFRDs7O0dBR0c7QUFDSCxNQUFhLGFBQWMsU0FBUSx5QkFBZ0I7SUFDakQ7Ozs7T0FJRztJQUNILE1BQU0sQ0FBQyxLQUFLLENBQUMsaUJBQWlCLENBQUMsT0FBNkI7UUFDMUQsTUFBTSxPQUFPLEdBQUcsTUFBTSw2Q0FBb0IsQ0FBQyxlQUFlLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDcEUsT0FBTyxJQUFJLGFBQWEsQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUNwQyxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILFlBQVksVUFBZ0M7UUFDMUMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxDQUFDO0lBQ3BCLENBQUM7SUFFRDs7O09BR0c7SUFDSCxLQUFLLENBQUMsSUFBSTtRQUNSLE1BQU0sSUFBSSxHQUFHLE1BQU0sSUFBSSxDQUFDLGVBQWUsRUFBRSxDQUFDO1FBQzFDLE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsSUFBQSxlQUFTLEVBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUN2QyxDQUFDO0NBQ0Y7QUE1QkQsc0NBNEJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ3ViZVNpZ25lckNsaWVudCB9IGZyb20gXCIuL2NsaWVudFwiO1xuaW1wb3J0IHsgS2V5SW5mbywgdG9LZXlJbmZvIH0gZnJvbSBcIi4va2V5XCI7XG5pbXBvcnQgeyBTaWduZXJTZXNzaW9uTWFuYWdlciwgU2lnbmVyU2Vzc2lvblN0b3JhZ2UgfSBmcm9tIFwiLi9zZXNzaW9uL3NpZ25lcl9zZXNzaW9uX21hbmFnZXJcIjtcblxuLyoqIFNpZ25lciBzZXNzaW9uIGluZm8uIENhbiBvbmx5IGJlIHVzZWQgdG8gcmV2b2tlIGEgdG9rZW4sIGJ1dCBub3QgZm9yIGF1dGhlbnRpY2F0aW9uLiAqL1xuZXhwb3J0IGNsYXNzIFNpZ25lclNlc3Npb25JbmZvIHtcbiAgcmVhZG9ubHkgI2NzYzogQ3ViZVNpZ25lckNsaWVudDtcbiAgcmVhZG9ubHkgI3Nlc3Npb25JZDogc3RyaW5nO1xuICBwdWJsaWMgcmVhZG9ubHkgcHVycG9zZTogc3RyaW5nO1xuXG4gIC8qKiBSZXZva2UgdGhpcyBzZXNzaW9uICovXG4gIGFzeW5jIHJldm9rZSgpIHtcbiAgICBhd2FpdCB0aGlzLiNjc2Muc2Vzc2lvblJldm9rZSh0aGlzLiNzZXNzaW9uSWQpO1xuICB9XG5cbiAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbiAgLy8gLS0gSU5URVJOQUwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbiAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cblxuICAvKipcbiAgICogSW50ZXJuYWwgY29uc3RydWN0b3IuXG4gICAqIEBwYXJhbSB7Q3ViZVNpZ25lckNsaWVudH0gY3MgQ3ViZVNpZ25lciBpbnN0YW5jZSB0byB1c2Ugd2hlbiBjYWxsaW5nIGByZXZva2VgXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBzZXNzaW9uSWQgVGhlIElEIG9mIHRoZSBzZXNzaW9uOyBjYW4gYmUgdXNlZCBmb3IgcmV2b2NhdGlvbiBidXQgbm90IGZvciBhdXRoXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBwdXJwb3NlIFNlc3Npb24gcHVycG9zZVxuICAgKiBAaW50ZXJuYWxcbiAgICovXG4gIGNvbnN0cnVjdG9yKGNzOiBDdWJlU2lnbmVyQ2xpZW50LCBzZXNzaW9uSWQ6IHN0cmluZywgcHVycG9zZTogc3RyaW5nKSB7XG4gICAgdGhpcy4jY3NjID0gY3M7XG4gICAgdGhpcy4jc2Vzc2lvbklkID0gc2Vzc2lvbklkO1xuICAgIHRoaXMucHVycG9zZSA9IHB1cnBvc2U7XG4gIH1cbn1cblxuLyoqXG4gKiBTaWduZXIgc2Vzc2lvbi5cbiAqIEV4dGVuZHMge0BsaW5rIEN1YmVTaWduZXJDbGllbnR9IGFuZCBwcm92aWRlcyBhIGZldyBjb252ZW5pZW5jZSBtZXRob2RzIG9uIHRvcC5cbiAqL1xuZXhwb3J0IGNsYXNzIFNpZ25lclNlc3Npb24gZXh0ZW5kcyBDdWJlU2lnbmVyQ2xpZW50IHtcbiAgLyoqXG4gICAqIExvYWRzIGFuIGV4aXN0aW5nIHNpZ25lciBzZXNzaW9uIGZyb20gc3RvcmFnZS5cbiAgICogQHBhcmFtIHtTaWduZXJTZXNzaW9uU3RvcmFnZX0gc3RvcmFnZSBUaGUgc2Vzc2lvbiBzdG9yYWdlIHRvIHVzZVxuICAgKiBAcmV0dXJuIHtQcm9taXNlPFNpbmdlclNlc3Npb24+fSBOZXcgc2lnbmVyIHNlc3Npb25cbiAgICovXG4gIHN0YXRpYyBhc3luYyBsb2FkU2lnbmVyU2Vzc2lvbihzdG9yYWdlOiBTaWduZXJTZXNzaW9uU3RvcmFnZSk6IFByb21pc2U8U2lnbmVyU2Vzc2lvbj4ge1xuICAgIGNvbnN0IG1hbmFnZXIgPSBhd2FpdCBTaWduZXJTZXNzaW9uTWFuYWdlci5sb2FkRnJvbVN0b3JhZ2Uoc3RvcmFnZSk7XG4gICAgcmV0dXJuIG5ldyBTaWduZXJTZXNzaW9uKG1hbmFnZXIpO1xuICB9XG5cbiAgLyoqXG4gICAqIENvbnN0cnVjdG9yLlxuICAgKiBAcGFyYW0ge1NpZ25lclNlc3Npb25NYW5hZ2VyfSBzZXNzaW9uTWdyIFRoZSBzZXNzaW9uIG1hbmFnZXIgdG8gdXNlXG4gICAqIEBpbnRlcm5hbFxuICAgKi9cbiAgY29uc3RydWN0b3Ioc2Vzc2lvbk1ncjogU2lnbmVyU2Vzc2lvbk1hbmFnZXIpIHtcbiAgICBzdXBlcihzZXNzaW9uTWdyKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIHRoZSBsaXN0IG9mIGtleXMgdGhhdCB0aGlzIHRva2VuIGdyYW50cyBhY2Nlc3MgdG8uXG4gICAqIEByZXR1cm4ge0tleUluZm9bXX0gVGhlIGxpc3Qgb2Yga2V5cy5cbiAgICovXG4gIGFzeW5jIGtleXMoKTogUHJvbWlzZTxLZXlJbmZvW10+IHtcbiAgICBjb25zdCBrZXlzID0gYXdhaXQgdGhpcy5zZXNzaW9uS2V5c0xpc3QoKTtcbiAgICByZXR1cm4ga2V5cy5tYXAoKGspID0+IHRvS2V5SW5mbyhrKSk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import { UserExportCompleteResponse, UserExportKeyMaterial } from "./schema_types";
|
|
2
|
+
import type { CipherSuite } from "@hpke/core";
|
|
3
|
+
/** Get the HPKE ciphersuite for user-export decryption.
|
|
4
|
+
*
|
|
5
|
+
* @return {any} The HPKE ciphersuite for user export.
|
|
6
|
+
*/
|
|
7
|
+
export declare function userExportCipherSuite(): Promise<CipherSuite>;
|
|
8
|
+
/**
|
|
9
|
+
* Generate a key pair for user export.
|
|
10
|
+
*
|
|
11
|
+
* @return {Promise<CryptoKeyPair>} The newly generated key pair.
|
|
12
|
+
*/
|
|
13
|
+
export declare function userExportKeygen(): Promise<CryptoKeyPair>;
|
|
14
|
+
/**
|
|
15
|
+
* Decrypt a user export.
|
|
16
|
+
*
|
|
17
|
+
* @param {CryptoKey} recipientKey The NIST P-256 secret key corresponding to the `publicKey` argument to the `userExportComplete` invocation that returned `response`.
|
|
18
|
+
* @param {UserExportCompleteResponse} response The response from a successful `userExportComplete` request.
|
|
19
|
+
* @return {Promise<UserExportKeyMaterial>} The decrypted key material.
|
|
20
|
+
*/
|
|
21
|
+
export declare function userExportDecrypt(recipientKey: CryptoKey, response: UserExportCompleteResponse): Promise<UserExportKeyMaterial>;
|
|
22
|
+
/**
|
|
23
|
+
* Figure out how to load SubtleCrypto in the current environment.
|
|
24
|
+
*
|
|
25
|
+
* This functionality is reproduced from the hpke-js package,
|
|
26
|
+
* https://github.com/dajiaji/hpke-js/
|
|
27
|
+
* which is Copyright (C) 2022 Ajitomi Daisuke and licensed
|
|
28
|
+
* under the MIT License, which follows:
|
|
29
|
+
*
|
|
30
|
+
* MIT License
|
|
31
|
+
*
|
|
32
|
+
* Copyright (c) 2022 Ajitomi Daisuke
|
|
33
|
+
*
|
|
34
|
+
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
35
|
+
* of this software and associated documentation files (the "Software"), to deal
|
|
36
|
+
* in the Software without restriction, including without limitation the rights
|
|
37
|
+
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
38
|
+
* copies of the Software, and to permit persons to whom the Software is
|
|
39
|
+
* furnished to do so, subject to the following conditions:
|
|
40
|
+
*
|
|
41
|
+
* The above copyright notice and this permission notice shall be included in all
|
|
42
|
+
* copies or substantial portions of the Software.
|
|
43
|
+
*
|
|
44
|
+
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
45
|
+
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
46
|
+
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
47
|
+
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
48
|
+
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
49
|
+
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
50
|
+
* SOFTWARE.
|
|
51
|
+
*/
|
|
52
|
+
export declare function loadSubtleCrypto(): Promise<SubtleCrypto>;
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || function (mod) {
|
|
19
|
+
if (mod && mod.__esModule) return mod;
|
|
20
|
+
var result = {};
|
|
21
|
+
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
22
|
+
__setModuleDefault(result, mod);
|
|
23
|
+
return result;
|
|
24
|
+
};
|
|
25
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
26
|
+
exports.loadSubtleCrypto = exports.userExportDecrypt = exports.userExportKeygen = exports.userExportCipherSuite = void 0;
|
|
27
|
+
const util_1 = require("./util");
|
|
28
|
+
/** Get the HPKE ciphersuite for user-export decryption.
|
|
29
|
+
*
|
|
30
|
+
* @return {any} The HPKE ciphersuite for user export.
|
|
31
|
+
*/
|
|
32
|
+
async function userExportCipherSuite() {
|
|
33
|
+
const hpke = await Promise.resolve().then(() => __importStar(require("@hpke/core"))); // eslint-disable-line @typescript-eslint/no-var-requires
|
|
34
|
+
const suite = new hpke.CipherSuite({
|
|
35
|
+
kem: new hpke.DhkemP256HkdfSha256(),
|
|
36
|
+
kdf: new hpke.HkdfSha256(),
|
|
37
|
+
aead: new hpke.Aes256Gcm(),
|
|
38
|
+
});
|
|
39
|
+
return suite;
|
|
40
|
+
}
|
|
41
|
+
exports.userExportCipherSuite = userExportCipherSuite;
|
|
42
|
+
/**
|
|
43
|
+
* Generate a key pair for user export.
|
|
44
|
+
*
|
|
45
|
+
* @return {Promise<CryptoKeyPair>} The newly generated key pair.
|
|
46
|
+
*/
|
|
47
|
+
async function userExportKeygen() {
|
|
48
|
+
return (await userExportCipherSuite()).kem.generateKeyPair();
|
|
49
|
+
}
|
|
50
|
+
exports.userExportKeygen = userExportKeygen;
|
|
51
|
+
/**
|
|
52
|
+
* Get the ArrayBuffer slice represented by a Buffer
|
|
53
|
+
*
|
|
54
|
+
* @param {Uint8Array} b The buffer to convert
|
|
55
|
+
* @return {ArrayBuffer} The resulting ArrayBuffer
|
|
56
|
+
*/
|
|
57
|
+
function toArrayBuffer(b) {
|
|
58
|
+
return b.buffer.slice(b.byteOffset, b.byteOffset + b.byteLength);
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Decrypt a user export.
|
|
62
|
+
*
|
|
63
|
+
* @param {CryptoKey} recipientKey The NIST P-256 secret key corresponding to the `publicKey` argument to the `userExportComplete` invocation that returned `response`.
|
|
64
|
+
* @param {UserExportCompleteResponse} response The response from a successful `userExportComplete` request.
|
|
65
|
+
* @return {Promise<UserExportKeyMaterial>} The decrypted key material.
|
|
66
|
+
*/
|
|
67
|
+
async function userExportDecrypt(recipientKey, response) {
|
|
68
|
+
// The ciphersuite we use for decryption
|
|
69
|
+
const suite = await userExportCipherSuite();
|
|
70
|
+
// decrypt the export ciphertext using the HPKE one-shot API
|
|
71
|
+
const tenc = new TextEncoder();
|
|
72
|
+
const tdec = new TextDecoder();
|
|
73
|
+
const info = toArrayBuffer(tenc.encode(`cubist-signer::UserExportOwner::${response.user_id}`));
|
|
74
|
+
const public_key = toArrayBuffer((0, util_1.decodeBase64)(response.ephemeral_public_key));
|
|
75
|
+
const ctxt = toArrayBuffer((0, util_1.decodeBase64)(response.encrypted_key_material));
|
|
76
|
+
const decrypted = JSON.parse(tdec.decode(await suite.open({
|
|
77
|
+
recipientKey,
|
|
78
|
+
enc: public_key,
|
|
79
|
+
info: info,
|
|
80
|
+
}, ctxt)));
|
|
81
|
+
return decrypted;
|
|
82
|
+
}
|
|
83
|
+
exports.userExportDecrypt = userExportDecrypt;
|
|
84
|
+
/**
|
|
85
|
+
* Figure out how to load SubtleCrypto in the current environment.
|
|
86
|
+
*
|
|
87
|
+
* This functionality is reproduced from the hpke-js package,
|
|
88
|
+
* https://github.com/dajiaji/hpke-js/
|
|
89
|
+
* which is Copyright (C) 2022 Ajitomi Daisuke and licensed
|
|
90
|
+
* under the MIT License, which follows:
|
|
91
|
+
*
|
|
92
|
+
* MIT License
|
|
93
|
+
*
|
|
94
|
+
* Copyright (c) 2022 Ajitomi Daisuke
|
|
95
|
+
*
|
|
96
|
+
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
97
|
+
* of this software and associated documentation files (the "Software"), to deal
|
|
98
|
+
* in the Software without restriction, including without limitation the rights
|
|
99
|
+
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
100
|
+
* copies of the Software, and to permit persons to whom the Software is
|
|
101
|
+
* furnished to do so, subject to the following conditions:
|
|
102
|
+
*
|
|
103
|
+
* The above copyright notice and this permission notice shall be included in all
|
|
104
|
+
* copies or substantial portions of the Software.
|
|
105
|
+
*
|
|
106
|
+
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
107
|
+
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
108
|
+
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
109
|
+
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
110
|
+
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
111
|
+
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
112
|
+
* SOFTWARE.
|
|
113
|
+
*/
|
|
114
|
+
async function loadSubtleCrypto() {
|
|
115
|
+
if (globalThis !== undefined && globalThis.crypto !== undefined) {
|
|
116
|
+
// Browsers, Node.js >= v19, Cloudflare Workers, Bun, etc.
|
|
117
|
+
return globalThis.crypto.subtle;
|
|
118
|
+
}
|
|
119
|
+
// Node.js <= v18
|
|
120
|
+
try {
|
|
121
|
+
const { webcrypto } = await Promise.resolve().then(() => __importStar(require("crypto"))); // node:crypto
|
|
122
|
+
return webcrypto.subtle;
|
|
123
|
+
}
|
|
124
|
+
catch (e) {
|
|
125
|
+
throw new Error("subtle crypto not supported");
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
exports.loadSubtleCrypto = loadSubtleCrypto;
|
|
129
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user_export.js","sourceRoot":"","sources":["../../src/user_export.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;AACA,iCAAsC;AAGtC;;;GAGG;AACI,KAAK,UAAU,qBAAqB;IACzC,MAAM,IAAI,GAAG,wDAAa,YAAY,GAAC,CAAC,CAAC,yDAAyD;IAClG,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,WAAW,CAAC;QACjC,GAAG,EAAE,IAAI,IAAI,CAAC,mBAAmB,EAAE;QACnC,GAAG,EAAE,IAAI,IAAI,CAAC,UAAU,EAAE;QAC1B,IAAI,EAAE,IAAI,IAAI,CAAC,SAAS,EAAE;KAC3B,CAAC,CAAC;IACH,OAAO,KAAK,CAAC;AACf,CAAC;AARD,sDAQC;AAED;;;;GAIG;AACI,KAAK,UAAU,gBAAgB;IACpC,OAAO,CAAC,MAAM,qBAAqB,EAAE,CAAC,CAAC,GAAG,CAAC,eAAe,EAAE,CAAC;AAC/D,CAAC;AAFD,4CAEC;AAED;;;;;GAKG;AACH,SAAS,aAAa,CAAC,CAAa;IAClC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,UAAU,EAAE,CAAC,CAAC,UAAU,GAAG,CAAC,CAAC,UAAU,CAAC,CAAC;AACnE,CAAC;AAED;;;;;;GAMG;AACI,KAAK,UAAU,iBAAiB,CACrC,YAAuB,EACvB,QAAoC;IAEpC,wCAAwC;IACxC,MAAM,KAAK,GAAG,MAAM,qBAAqB,EAAE,CAAC;IAE5C,4DAA4D;IAC5D,MAAM,IAAI,GAAG,IAAI,WAAW,EAAE,CAAC;IAC/B,MAAM,IAAI,GAAG,IAAI,WAAW,EAAE,CAAC;IAC/B,MAAM,IAAI,GAAG,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,mCAAmC,QAAQ,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;IAC/F,MAAM,UAAU,GAAG,aAAa,CAAC,IAAA,mBAAY,EAAC,QAAQ,CAAC,oBAAoB,CAAC,CAAC,CAAC;IAC9E,MAAM,IAAI,GAAG,aAAa,CAAC,IAAA,mBAAY,EAAC,QAAQ,CAAC,sBAAsB,CAAC,CAAC,CAAC;IAC1E,MAAM,SAAS,GAA0B,IAAI,CAAC,KAAK,CACjD,IAAI,CAAC,MAAM,CACT,MAAM,KAAK,CAAC,IAAI,CACd;QACE,YAAY;QACZ,GAAG,EAAE,UAAU;QACf,IAAI,EAAE,IAAI;KACX,EACD,IAAI,CACL,CACF,CACF,CAAC;IAEF,OAAO,SAAS,CAAC;AACnB,CAAC;AA3BD,8CA2BC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6BG;AACI,KAAK,UAAU,gBAAgB;IACpC,IAAI,UAAU,KAAK,SAAS,IAAI,UAAU,CAAC,MAAM,KAAK,SAAS,EAAE,CAAC;QAChE,0DAA0D;QAC1D,OAAO,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC;IAClC,CAAC;IACD,iBAAiB;IACjB,IAAI,CAAC;QACH,MAAM,EAAE,SAAS,EAAE,GAAG,wDAAa,QAAQ,GAAC,CAAC,CAAC,cAAc;QAC5D,OAAQ,SAA+B,CAAC,MAAM,CAAC;IACjD,CAAC;IAAC,OAAO,CAAU,EAAE,CAAC;QACpB,MAAM,IAAI,KAAK,CAAC,6BAA6B,CAAC,CAAC;IACjD,CAAC;AACH,CAAC;AAZD,4CAYC","sourcesContent":["import { UserExportCompleteResponse, UserExportKeyMaterial } from \"./schema_types\";\nimport { decodeBase64 } from \"./util\";\nimport type { CipherSuite } from \"@hpke/core\";\n\n/** Get the HPKE ciphersuite for user-export decryption.\n *\n * @return {any} The HPKE ciphersuite for user export.\n */\nexport async function userExportCipherSuite(): Promise<CipherSuite> {\n  const hpke = await import(\"@hpke/core\"); // eslint-disable-line @typescript-eslint/no-var-requires\n  const suite = new hpke.CipherSuite({\n    kem: new hpke.DhkemP256HkdfSha256(),\n    kdf: new hpke.HkdfSha256(),\n    aead: new hpke.Aes256Gcm(),\n  });\n  return suite;\n}\n\n/**\n * Generate a key pair for user export.\n *\n * @return {Promise<CryptoKeyPair>} The newly generated key pair.\n */\nexport async function userExportKeygen(): Promise<CryptoKeyPair> {\n  return (await userExportCipherSuite()).kem.generateKeyPair();\n}\n\n/**\n * Get the ArrayBuffer slice represented by a Buffer\n *\n * @param {Uint8Array} b The buffer to convert\n * @return {ArrayBuffer} The resulting ArrayBuffer\n */\nfunction toArrayBuffer(b: Uint8Array): ArrayBuffer {\n  return b.buffer.slice(b.byteOffset, b.byteOffset + b.byteLength);\n}\n\n/**\n * Decrypt a user export.\n *\n * @param {CryptoKey} recipientKey The NIST P-256 secret key corresponding to the `publicKey` argument to the `userExportComplete` invocation that returned `response`.\n * @param {UserExportCompleteResponse} response The response from a successful `userExportComplete` request.\n * @return {Promise<UserExportKeyMaterial>} The decrypted key material.\n */\nexport async function userExportDecrypt(\n  recipientKey: CryptoKey,\n  response: UserExportCompleteResponse,\n): Promise<UserExportKeyMaterial> {\n  // The ciphersuite we use for decryption\n  const suite = await userExportCipherSuite();\n\n  // decrypt the export ciphertext using the HPKE one-shot API\n  const tenc = new TextEncoder();\n  const tdec = new TextDecoder();\n  const info = toArrayBuffer(tenc.encode(`cubist-signer::UserExportOwner::${response.user_id}`));\n  const public_key = toArrayBuffer(decodeBase64(response.ephemeral_public_key));\n  const ctxt = toArrayBuffer(decodeBase64(response.encrypted_key_material));\n  const decrypted: UserExportKeyMaterial = JSON.parse(\n    tdec.decode(\n      await suite.open(\n        {\n          recipientKey,\n          enc: public_key,\n          info: info,\n        },\n        ctxt,\n      ),\n    ),\n  );\n\n  return decrypted;\n}\n\n/**\n * Figure out how to load SubtleCrypto in the current environment.\n *\n * This functionality is reproduced from the hpke-js package,\n *   https://github.com/dajiaji/hpke-js/\n * which is Copyright (C) 2022 Ajitomi Daisuke and licensed\n * under the MIT License, which follows:\n *\n * MIT License\n *\n * Copyright (c) 2022 Ajitomi Daisuke\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\nexport async function loadSubtleCrypto() {\n  if (globalThis !== undefined && globalThis.crypto !== undefined) {\n    // Browsers, Node.js >= v19, Cloudflare Workers, Bun, etc.\n    return globalThis.crypto.subtle;\n  }\n  // Node.js <= v18\n  try {\n    const { webcrypto } = await import(\"crypto\"); // node:crypto\n    return (webcrypto as unknown as Crypto).subtle;\n  } catch (e: unknown) {\n    throw new Error(\"subtle crypto not supported\");\n  }\n}\n"]}
|