@nitrostack/core 1.0.0
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/LICENSE +201 -0
- package/README.md +80 -0
- package/dist/auth/api-key.d.ts +118 -0
- package/dist/auth/api-key.d.ts.map +1 -0
- package/dist/auth/api-key.js +168 -0
- package/dist/auth/api-key.js.map +1 -0
- package/dist/auth/client.d.ts +151 -0
- package/dist/auth/client.d.ts.map +1 -0
- package/dist/auth/client.js +330 -0
- package/dist/auth/client.js.map +1 -0
- package/dist/auth/index.d.ts +31 -0
- package/dist/auth/index.d.ts.map +1 -0
- package/dist/auth/index.js +46 -0
- package/dist/auth/index.js.map +1 -0
- package/dist/auth/middleware.d.ts +95 -0
- package/dist/auth/middleware.d.ts.map +1 -0
- package/dist/auth/middleware.js +260 -0
- package/dist/auth/middleware.js.map +1 -0
- package/dist/auth/pkce.d.ts +53 -0
- package/dist/auth/pkce.d.ts.map +1 -0
- package/dist/auth/pkce.js +105 -0
- package/dist/auth/pkce.js.map +1 -0
- package/dist/auth/quick-setup.d.ts +94 -0
- package/dist/auth/quick-setup.d.ts.map +1 -0
- package/dist/auth/quick-setup.js +210 -0
- package/dist/auth/quick-setup.js.map +1 -0
- package/dist/auth/secure-secret.d.ts +136 -0
- package/dist/auth/secure-secret.d.ts.map +1 -0
- package/dist/auth/secure-secret.js +182 -0
- package/dist/auth/secure-secret.js.map +1 -0
- package/dist/auth/server-integration.d.ts +97 -0
- package/dist/auth/server-integration.d.ts.map +1 -0
- package/dist/auth/server-integration.js +182 -0
- package/dist/auth/server-integration.js.map +1 -0
- package/dist/auth/server-metadata.d.ts +51 -0
- package/dist/auth/server-metadata.d.ts.map +1 -0
- package/dist/auth/server-metadata.js +106 -0
- package/dist/auth/server-metadata.js.map +1 -0
- package/dist/auth/simple-jwt.d.ts +174 -0
- package/dist/auth/simple-jwt.d.ts.map +1 -0
- package/dist/auth/simple-jwt.js +162 -0
- package/dist/auth/simple-jwt.js.map +1 -0
- package/dist/auth/token-store.d.ts +104 -0
- package/dist/auth/token-store.d.ts.map +1 -0
- package/dist/auth/token-store.js +205 -0
- package/dist/auth/token-store.js.map +1 -0
- package/dist/auth/token-validation.d.ts +59 -0
- package/dist/auth/token-validation.d.ts.map +1 -0
- package/dist/auth/token-validation.js +241 -0
- package/dist/auth/token-validation.js.map +1 -0
- package/dist/auth/types.d.ts +215 -0
- package/dist/auth/types.d.ts.map +1 -0
- package/dist/auth/types.js +6 -0
- package/dist/auth/types.js.map +1 -0
- package/dist/core/apikey-module.d.ts +69 -0
- package/dist/core/apikey-module.d.ts.map +1 -0
- package/dist/core/apikey-module.js +114 -0
- package/dist/core/apikey-module.js.map +1 -0
- package/dist/core/app-decorator.d.ts +59 -0
- package/dist/core/app-decorator.d.ts.map +1 -0
- package/dist/core/app-decorator.js +322 -0
- package/dist/core/app-decorator.js.map +1 -0
- package/dist/core/builders.d.ts +50 -0
- package/dist/core/builders.d.ts.map +1 -0
- package/dist/core/builders.js +139 -0
- package/dist/core/builders.js.map +1 -0
- package/dist/core/component.d.ts +111 -0
- package/dist/core/component.d.ts.map +1 -0
- package/dist/core/component.js +228 -0
- package/dist/core/component.js.map +1 -0
- package/dist/core/config-module.d.ts +62 -0
- package/dist/core/config-module.d.ts.map +1 -0
- package/dist/core/config-module.js +94 -0
- package/dist/core/config-module.js.map +1 -0
- package/dist/core/decorators/cache.decorator.d.ts +61 -0
- package/dist/core/decorators/cache.decorator.d.ts.map +1 -0
- package/dist/core/decorators/cache.decorator.js +115 -0
- package/dist/core/decorators/cache.decorator.js.map +1 -0
- package/dist/core/decorators/health-check.decorator.d.ts +80 -0
- package/dist/core/decorators/health-check.decorator.d.ts.map +1 -0
- package/dist/core/decorators/health-check.decorator.js +153 -0
- package/dist/core/decorators/health-check.decorator.js.map +1 -0
- package/dist/core/decorators/rate-limit.decorator.d.ts +63 -0
- package/dist/core/decorators/rate-limit.decorator.d.ts.map +1 -0
- package/dist/core/decorators/rate-limit.decorator.js +129 -0
- package/dist/core/decorators/rate-limit.decorator.js.map +1 -0
- package/dist/core/decorators.d.ts +190 -0
- package/dist/core/decorators.d.ts.map +1 -0
- package/dist/core/decorators.js +170 -0
- package/dist/core/decorators.js.map +1 -0
- package/dist/core/di/container.d.ts +64 -0
- package/dist/core/di/container.d.ts.map +1 -0
- package/dist/core/di/container.js +105 -0
- package/dist/core/di/container.js.map +1 -0
- package/dist/core/di/injectable.decorator.d.ts +62 -0
- package/dist/core/di/injectable.decorator.d.ts.map +1 -0
- package/dist/core/di/injectable.decorator.js +66 -0
- package/dist/core/di/injectable.decorator.js.map +1 -0
- package/dist/core/errors.d.ts +54 -0
- package/dist/core/errors.d.ts.map +1 -0
- package/dist/core/errors.js +87 -0
- package/dist/core/errors.js.map +1 -0
- package/dist/core/events/event-emitter.d.ts +50 -0
- package/dist/core/events/event-emitter.d.ts.map +1 -0
- package/dist/core/events/event-emitter.js +94 -0
- package/dist/core/events/event-emitter.js.map +1 -0
- package/dist/core/events/event.decorator.d.ts +48 -0
- package/dist/core/events/event.decorator.d.ts.map +1 -0
- package/dist/core/events/event.decorator.js +72 -0
- package/dist/core/events/event.decorator.js.map +1 -0
- package/dist/core/events/log-emitter.d.ts +14 -0
- package/dist/core/events/log-emitter.d.ts.map +1 -0
- package/dist/core/events/log-emitter.js +20 -0
- package/dist/core/events/log-emitter.js.map +1 -0
- package/dist/core/filters/exception-filter.decorator.d.ts +40 -0
- package/dist/core/filters/exception-filter.decorator.d.ts.map +1 -0
- package/dist/core/filters/exception-filter.decorator.js +54 -0
- package/dist/core/filters/exception-filter.decorator.js.map +1 -0
- package/dist/core/filters/exception-filter.interface.d.ts +39 -0
- package/dist/core/filters/exception-filter.interface.d.ts.map +1 -0
- package/dist/core/filters/exception-filter.interface.js +2 -0
- package/dist/core/filters/exception-filter.interface.js.map +1 -0
- package/dist/core/guards/apikey.guard.d.ts +22 -0
- package/dist/core/guards/apikey.guard.d.ts.map +1 -0
- package/dist/core/guards/apikey.guard.js +11 -0
- package/dist/core/guards/apikey.guard.js.map +1 -0
- package/dist/core/guards/guard.interface.d.ts +18 -0
- package/dist/core/guards/guard.interface.d.ts.map +1 -0
- package/dist/core/guards/guard.interface.js +2 -0
- package/dist/core/guards/guard.interface.js.map +1 -0
- package/dist/core/guards/jwt.guard.d.ts +18 -0
- package/dist/core/guards/jwt.guard.d.ts.map +1 -0
- package/dist/core/guards/jwt.guard.js +2 -0
- package/dist/core/guards/jwt.guard.js.map +1 -0
- package/dist/core/guards/oauth.guard.d.ts +35 -0
- package/dist/core/guards/oauth.guard.d.ts.map +1 -0
- package/dist/core/guards/oauth.guard.js +2 -0
- package/dist/core/guards/oauth.guard.js.map +1 -0
- package/dist/core/guards/use-guards.decorator.d.ts +25 -0
- package/dist/core/guards/use-guards.decorator.d.ts.map +1 -0
- package/dist/core/guards/use-guards.decorator.js +32 -0
- package/dist/core/guards/use-guards.decorator.js.map +1 -0
- package/dist/core/health/health-checks.resource.d.ts +14 -0
- package/dist/core/health/health-checks.resource.d.ts.map +1 -0
- package/dist/core/health/health-checks.resource.js +29 -0
- package/dist/core/health/health-checks.resource.js.map +1 -0
- package/dist/core/index.d.ts +57 -0
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +59 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/interceptors/interceptor.decorator.d.ts +37 -0
- package/dist/core/interceptors/interceptor.decorator.d.ts.map +1 -0
- package/dist/core/interceptors/interceptor.decorator.js +51 -0
- package/dist/core/interceptors/interceptor.decorator.js.map +1 -0
- package/dist/core/interceptors/interceptor.interface.d.ts +31 -0
- package/dist/core/interceptors/interceptor.interface.d.ts.map +1 -0
- package/dist/core/interceptors/interceptor.interface.js +2 -0
- package/dist/core/interceptors/interceptor.interface.js.map +1 -0
- package/dist/core/jwt-module.d.ts +51 -0
- package/dist/core/jwt-module.d.ts.map +1 -0
- package/dist/core/jwt-module.js +52 -0
- package/dist/core/jwt-module.js.map +1 -0
- package/dist/core/logger.d.ts +18 -0
- package/dist/core/logger.d.ts.map +1 -0
- package/dist/core/logger.js +53 -0
- package/dist/core/logger.js.map +1 -0
- package/dist/core/middleware/middleware.decorator.d.ts +39 -0
- package/dist/core/middleware/middleware.decorator.d.ts.map +1 -0
- package/dist/core/middleware/middleware.decorator.js +53 -0
- package/dist/core/middleware/middleware.decorator.js.map +1 -0
- package/dist/core/middleware/middleware.interface.d.ts +29 -0
- package/dist/core/middleware/middleware.interface.d.ts.map +1 -0
- package/dist/core/middleware/middleware.interface.js +2 -0
- package/dist/core/middleware/middleware.interface.js.map +1 -0
- package/dist/core/module.d.ts +93 -0
- package/dist/core/module.d.ts.map +1 -0
- package/dist/core/module.js +87 -0
- package/dist/core/module.js.map +1 -0
- package/dist/core/oauth-module.d.ts +123 -0
- package/dist/core/oauth-module.d.ts.map +1 -0
- package/dist/core/oauth-module.js +324 -0
- package/dist/core/oauth-module.js.map +1 -0
- package/dist/core/pipes/pipe.decorator.d.ts +64 -0
- package/dist/core/pipes/pipe.decorator.d.ts.map +1 -0
- package/dist/core/pipes/pipe.decorator.js +85 -0
- package/dist/core/pipes/pipe.decorator.js.map +1 -0
- package/dist/core/pipes/pipe.interface.d.ts +41 -0
- package/dist/core/pipes/pipe.interface.d.ts.map +1 -0
- package/dist/core/pipes/pipe.interface.js +2 -0
- package/dist/core/pipes/pipe.interface.js.map +1 -0
- package/dist/core/prompt.d.ts +46 -0
- package/dist/core/prompt.d.ts.map +1 -0
- package/dist/core/prompt.js +76 -0
- package/dist/core/prompt.js.map +1 -0
- package/dist/core/resource.d.ts +47 -0
- package/dist/core/resource.d.ts.map +1 -0
- package/dist/core/resource.js +90 -0
- package/dist/core/resource.js.map +1 -0
- package/dist/core/server.d.ts +129 -0
- package/dist/core/server.d.ts.map +1 -0
- package/dist/core/server.js +617 -0
- package/dist/core/server.js.map +1 -0
- package/dist/core/tool.d.ts +108 -0
- package/dist/core/tool.d.ts.map +1 -0
- package/dist/core/tool.js +241 -0
- package/dist/core/tool.js.map +1 -0
- package/dist/core/transports/discovery-http-server.d.ts +19 -0
- package/dist/core/transports/discovery-http-server.d.ts.map +1 -0
- package/dist/core/transports/discovery-http-server.js +54 -0
- package/dist/core/transports/discovery-http-server.js.map +1 -0
- package/dist/core/transports/http-server.d.ts +108 -0
- package/dist/core/transports/http-server.d.ts.map +1 -0
- package/dist/core/transports/http-server.js +293 -0
- package/dist/core/transports/http-server.js.map +1 -0
- package/dist/core/transports/streamable-http.d.ts +177 -0
- package/dist/core/transports/streamable-http.d.ts.map +1 -0
- package/dist/core/transports/streamable-http.js +1287 -0
- package/dist/core/transports/streamable-http.js.map +1 -0
- package/dist/core/types.d.ts +195 -0
- package/dist/core/types.d.ts.map +1 -0
- package/dist/core/types.js +2 -0
- package/dist/core/types.js.map +1 -0
- package/dist/core/widgets/widget-examples.resource.d.ts +17 -0
- package/dist/core/widgets/widget-examples.resource.d.ts.map +1 -0
- package/dist/core/widgets/widget-examples.resource.js +28 -0
- package/dist/core/widgets/widget-examples.resource.js.map +1 -0
- package/dist/core/widgets/widget-registry.d.ts +56 -0
- package/dist/core/widgets/widget-registry.d.ts.map +1 -0
- package/dist/core/widgets/widget-registry.js +75 -0
- package/dist/core/widgets/widget-registry.js.map +1 -0
- package/dist/testing/index.d.ts +103 -0
- package/dist/testing/index.d.ts.map +1 -0
- package/dist/testing/index.js +161 -0
- package/dist/testing/index.js.map +1 -0
- package/dist/ui-next/index.d.ts +31 -0
- package/dist/ui-next/index.d.ts.map +1 -0
- package/dist/ui-next/index.js +687 -0
- package/dist/ui-next/index.js.map +1 -0
- package/package.json +89 -0
|
@@ -0,0 +1,162 @@
|
|
|
1
|
+
import jwt from 'jsonwebtoken';
|
|
2
|
+
import { unwrapSecret } from './secure-secret.js';
|
|
3
|
+
/**
|
|
4
|
+
* Create Simple JWT authentication middleware
|
|
5
|
+
*
|
|
6
|
+
* @example
|
|
7
|
+
* ```typescript
|
|
8
|
+
* const server = createServer({...});
|
|
9
|
+
*
|
|
10
|
+
* // Simple JWT auth (no OAuth complexity!)
|
|
11
|
+
* server.app.use('/mcp', createSimpleJWTAuth({
|
|
12
|
+
* secret: process.env.JWT_SECRET!,
|
|
13
|
+
* audience: 'my-mcp-server',
|
|
14
|
+
* issuer: 'my-app',
|
|
15
|
+
* }));
|
|
16
|
+
*
|
|
17
|
+
* server.start();
|
|
18
|
+
* ```
|
|
19
|
+
*/
|
|
20
|
+
export function createSimpleJWTAuth(config) {
|
|
21
|
+
const algorithm = config.algorithm || 'HS256';
|
|
22
|
+
return async (req, res, next) => {
|
|
23
|
+
try {
|
|
24
|
+
// 1. Extract token from Authorization header
|
|
25
|
+
const authHeader = req.headers.authorization;
|
|
26
|
+
if (!authHeader || !authHeader.startsWith('Bearer ')) {
|
|
27
|
+
return res.status(401).json({
|
|
28
|
+
error: 'unauthorized',
|
|
29
|
+
message: 'Missing or invalid Authorization header. Use: Authorization: Bearer <token>',
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
const token = authHeader.substring(7); // Remove 'Bearer '
|
|
33
|
+
// 2. Verify JWT
|
|
34
|
+
const verifyOptions = {
|
|
35
|
+
algorithms: [algorithm],
|
|
36
|
+
};
|
|
37
|
+
if (config.audience) {
|
|
38
|
+
verifyOptions.audience = config.audience;
|
|
39
|
+
}
|
|
40
|
+
if (config.issuer) {
|
|
41
|
+
verifyOptions.issuer = config.issuer;
|
|
42
|
+
}
|
|
43
|
+
const secretValue = unwrapSecret(config.secret);
|
|
44
|
+
const payload = jwt.verify(token, secretValue, verifyOptions);
|
|
45
|
+
// 3. Custom validation
|
|
46
|
+
if (config.customValidation && !config.customValidation(payload)) {
|
|
47
|
+
return res.status(403).json({
|
|
48
|
+
error: 'forbidden',
|
|
49
|
+
message: 'Token validation failed',
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
// 4. Attach to request
|
|
53
|
+
req.auth = {
|
|
54
|
+
authenticated: true,
|
|
55
|
+
tokenInfo: {
|
|
56
|
+
active: true,
|
|
57
|
+
sub: payload.sub,
|
|
58
|
+
aud: typeof payload.aud === 'string' ? [payload.aud] : payload.aud,
|
|
59
|
+
iss: payload.iss,
|
|
60
|
+
exp: payload.exp,
|
|
61
|
+
iat: payload.iat,
|
|
62
|
+
},
|
|
63
|
+
scopes: payload.scopes || payload.scope?.split(' ') || [],
|
|
64
|
+
clientId: payload.client_id || payload.sub,
|
|
65
|
+
subject: payload.sub,
|
|
66
|
+
};
|
|
67
|
+
next();
|
|
68
|
+
}
|
|
69
|
+
catch (error) {
|
|
70
|
+
const err = error;
|
|
71
|
+
if (err.name === 'TokenExpiredError') {
|
|
72
|
+
return res.status(401).json({
|
|
73
|
+
error: 'token_expired',
|
|
74
|
+
message: 'JWT has expired',
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
if (err.name === 'JsonWebTokenError') {
|
|
78
|
+
return res.status(401).json({
|
|
79
|
+
error: 'invalid_token',
|
|
80
|
+
message: 'Invalid JWT: ' + err.message,
|
|
81
|
+
});
|
|
82
|
+
}
|
|
83
|
+
return res.status(500).json({
|
|
84
|
+
error: 'server_error',
|
|
85
|
+
message: 'Token validation failed',
|
|
86
|
+
});
|
|
87
|
+
}
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* Generate a JWT token
|
|
92
|
+
*
|
|
93
|
+
* @example
|
|
94
|
+
* ```typescript
|
|
95
|
+
* const token = generateJWT({
|
|
96
|
+
* secret: SecretValue.fromEnv('JWT_SECRET'),
|
|
97
|
+
* payload: {
|
|
98
|
+
* sub: 'user123',
|
|
99
|
+
* scopes: ['mcp:read', 'mcp:write'],
|
|
100
|
+
* },
|
|
101
|
+
* expiresIn: '1h',
|
|
102
|
+
* });
|
|
103
|
+
* ```
|
|
104
|
+
*/
|
|
105
|
+
export function generateJWT(options) {
|
|
106
|
+
const signOptions = {
|
|
107
|
+
algorithm: options.algorithm || 'HS256',
|
|
108
|
+
};
|
|
109
|
+
if (options.expiresIn !== undefined) {
|
|
110
|
+
// Cast to jwt.SignOptions['expiresIn'] which accepts string | number
|
|
111
|
+
signOptions.expiresIn = options.expiresIn;
|
|
112
|
+
}
|
|
113
|
+
if (options.audience) {
|
|
114
|
+
signOptions.audience = options.audience;
|
|
115
|
+
}
|
|
116
|
+
if (options.issuer) {
|
|
117
|
+
signOptions.issuer = options.issuer;
|
|
118
|
+
}
|
|
119
|
+
const secretValue = unwrapSecret(options.secret);
|
|
120
|
+
return jwt.sign(options.payload, secretValue, signOptions);
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Verify a JWT token without middleware (helper)
|
|
124
|
+
*
|
|
125
|
+
* @param token - The JWT token to verify
|
|
126
|
+
* @param config - JWT configuration including secret
|
|
127
|
+
* @returns The decoded payload if valid, null otherwise
|
|
128
|
+
*/
|
|
129
|
+
export function verifyJWT(token, config) {
|
|
130
|
+
try {
|
|
131
|
+
const verifyOptions = {
|
|
132
|
+
algorithms: [config.algorithm || 'HS256'],
|
|
133
|
+
};
|
|
134
|
+
if (config.audience) {
|
|
135
|
+
verifyOptions.audience = config.audience;
|
|
136
|
+
}
|
|
137
|
+
if (config.issuer) {
|
|
138
|
+
verifyOptions.issuer = config.issuer;
|
|
139
|
+
}
|
|
140
|
+
const secretValue = unwrapSecret(config.secret);
|
|
141
|
+
const payload = jwt.verify(token, secretValue, verifyOptions);
|
|
142
|
+
if (config.customValidation && !config.customValidation(payload)) {
|
|
143
|
+
return null;
|
|
144
|
+
}
|
|
145
|
+
return payload;
|
|
146
|
+
}
|
|
147
|
+
catch {
|
|
148
|
+
return null;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* Decode JWT without verification (for debugging)
|
|
153
|
+
*/
|
|
154
|
+
export function decodeJWT(token) {
|
|
155
|
+
try {
|
|
156
|
+
return jwt.decode(token);
|
|
157
|
+
}
|
|
158
|
+
catch {
|
|
159
|
+
return null;
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
//# sourceMappingURL=simple-jwt.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"simple-jwt.js","sourceRoot":"","sources":["../../src/auth/simple-jwt.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import { StoredToken } from './types.js';
|
|
2
|
+
/**
|
|
3
|
+
* Token Storage Interface
|
|
4
|
+
*
|
|
5
|
+
* Provides secure storage for OAuth tokens
|
|
6
|
+
*/
|
|
7
|
+
export interface TokenStore {
|
|
8
|
+
/**
|
|
9
|
+
* Save a token
|
|
10
|
+
*/
|
|
11
|
+
saveToken(key: string, token: StoredToken): Promise<void>;
|
|
12
|
+
/**
|
|
13
|
+
* Get a token
|
|
14
|
+
*/
|
|
15
|
+
getToken(key: string): Promise<StoredToken | null>;
|
|
16
|
+
/**
|
|
17
|
+
* Delete a token
|
|
18
|
+
*/
|
|
19
|
+
deleteToken(key: string): Promise<void>;
|
|
20
|
+
/**
|
|
21
|
+
* List all stored token keys
|
|
22
|
+
*/
|
|
23
|
+
listKeys(): Promise<string[]>;
|
|
24
|
+
/**
|
|
25
|
+
* Clear all tokens
|
|
26
|
+
*/
|
|
27
|
+
clear(): Promise<void>;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* In-memory token store
|
|
31
|
+
* For testing and ephemeral storage
|
|
32
|
+
*/
|
|
33
|
+
export declare class MemoryTokenStore implements TokenStore {
|
|
34
|
+
private tokens;
|
|
35
|
+
saveToken(key: string, token: StoredToken): Promise<void>;
|
|
36
|
+
getToken(key: string): Promise<StoredToken | null>;
|
|
37
|
+
deleteToken(key: string): Promise<void>;
|
|
38
|
+
listKeys(): Promise<string[]>;
|
|
39
|
+
clear(): Promise<void>;
|
|
40
|
+
private isTokenExpired;
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* File-based token store with encryption
|
|
44
|
+
* For CLI applications and development
|
|
45
|
+
*/
|
|
46
|
+
export declare class FileTokenStore implements TokenStore {
|
|
47
|
+
private storePath;
|
|
48
|
+
private encryptionKey?;
|
|
49
|
+
constructor(storePath: string, encryptionKey?: string);
|
|
50
|
+
saveToken(key: string, token: StoredToken): Promise<void>;
|
|
51
|
+
getToken(key: string): Promise<StoredToken | null>;
|
|
52
|
+
deleteToken(key: string): Promise<void>;
|
|
53
|
+
listKeys(): Promise<string[]>;
|
|
54
|
+
clear(): Promise<void>;
|
|
55
|
+
/**
|
|
56
|
+
* Load tokens from file
|
|
57
|
+
*/
|
|
58
|
+
private loadTokens;
|
|
59
|
+
/**
|
|
60
|
+
* Save tokens to file
|
|
61
|
+
*/
|
|
62
|
+
private saveTokens;
|
|
63
|
+
/**
|
|
64
|
+
* Encrypt data using AES-256-GCM
|
|
65
|
+
*/
|
|
66
|
+
private encrypt;
|
|
67
|
+
/**
|
|
68
|
+
* Decrypt data using AES-256-GCM
|
|
69
|
+
*/
|
|
70
|
+
private decrypt;
|
|
71
|
+
/**
|
|
72
|
+
* Derive 256-bit key from password using PBKDF2
|
|
73
|
+
*/
|
|
74
|
+
private deriveKey;
|
|
75
|
+
private isTokenExpired;
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Create default token store
|
|
79
|
+
*
|
|
80
|
+
* Uses file-based storage with encryption if password provided
|
|
81
|
+
*
|
|
82
|
+
* @param storePath - Path to store tokens (default: ~/.nitrostack/tokens.json)
|
|
83
|
+
* @param encryptionKey - Optional encryption key
|
|
84
|
+
*/
|
|
85
|
+
export declare function createDefaultTokenStore(storePath?: string, encryptionKey?: string): TokenStore;
|
|
86
|
+
/**
|
|
87
|
+
* Utility: Check if token is expired
|
|
88
|
+
*/
|
|
89
|
+
export declare function isTokenExpired(token: StoredToken): boolean;
|
|
90
|
+
/**
|
|
91
|
+
* Utility: Calculate token expiration timestamp
|
|
92
|
+
*/
|
|
93
|
+
export declare function calculateExpiration(expiresIn: number): number;
|
|
94
|
+
/**
|
|
95
|
+
* Utility: Convert TokenResponse to StoredToken
|
|
96
|
+
*/
|
|
97
|
+
export declare function tokenResponseToStored(response: {
|
|
98
|
+
access_token: string;
|
|
99
|
+
token_type: 'Bearer';
|
|
100
|
+
expires_in?: number;
|
|
101
|
+
refresh_token?: string;
|
|
102
|
+
scope?: string;
|
|
103
|
+
}, resource?: string): StoredToken;
|
|
104
|
+
//# sourceMappingURL=token-store.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"token-store.d.ts","sourceRoot":"","sources":["../../src/auth/token-store.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAEzC;;;;GAIG;AACH,MAAM,WAAW,UAAU;IACzB;;OAEG;IACH,SAAS,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAE1D;;OAEG;IACH,QAAQ,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,WAAW,GAAG,IAAI,CAAC,CAAC;IAEnD;;OAEG;IACH,WAAW,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;IAExC;;OAEG;IACH,QAAQ,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;IAE9B;;OAEG;IACH,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC;CACxB;AAED;;;GAGG;AACH,qBAAa,gBAAiB,YAAW,UAAU;IACjD,OAAO,CAAC,MAAM,CAAuC;IAE/C,SAAS,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;IAIzD,QAAQ,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,WAAW,GAAG,IAAI,CAAC;IAalD,WAAW,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAIvC,QAAQ,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;IAI7B,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAI5B,OAAO,CAAC,cAAc;CAGvB;AAED;;;GAGG;AACH,qBAAa,cAAe,YAAW,UAAU;IAC/C,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,aAAa,CAAC,CAAS;gBAEnB,SAAS,EAAE,MAAM,EAAE,aAAa,CAAC,EAAE,MAAM;IAK/C,SAAS,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;IAMzD,QAAQ,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,WAAW,GAAG,IAAI,CAAC;IAgBlD,WAAW,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAMvC,QAAQ,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;IAK7B,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAI5B;;OAEG;YACW,UAAU;IAoBxB;;OAEG;YACW,UAAU;IAYxB;;OAEG;IACH,OAAO,CAAC,OAAO;IAgBf;;OAEG;IACH,OAAO,CAAC,OAAO;IAuBf;;OAEG;IACH,OAAO,CAAC,SAAS;IAOjB,OAAO,CAAC,cAAc;CAGvB;AAED;;;;;;;GAOG;AACH,wBAAgB,uBAAuB,CACrC,SAAS,CAAC,EAAE,MAAM,EAClB,aAAa,CAAC,EAAE,MAAM,GACrB,UAAU,CAQZ;AAUD;;GAEG;AACH,wBAAgB,cAAc,CAAC,KAAK,EAAE,WAAW,GAAG,OAAO,CAE1D;AAED;;GAEG;AACH,wBAAgB,mBAAmB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,CAE7D;AAED;;GAEG;AACH,wBAAgB,qBAAqB,CACnC,QAAQ,EAAE;IAAE,YAAY,EAAE,MAAM,CAAC;IAAC,UAAU,EAAE,QAAQ,CAAC;IAAC,UAAU,CAAC,EAAE,MAAM,CAAC;IAAC,aAAa,CAAC,EAAE,MAAM,CAAC;IAAC,KAAK,CAAC,EAAE,MAAM,CAAA;CAAE,EACrH,QAAQ,CAAC,EAAE,MAAM,GAChB,WAAW,CASb"}
|
|
@@ -0,0 +1,205 @@
|
|
|
1
|
+
import fs from 'fs/promises';
|
|
2
|
+
import path from 'path';
|
|
3
|
+
import crypto from 'crypto';
|
|
4
|
+
/**
|
|
5
|
+
* In-memory token store
|
|
6
|
+
* For testing and ephemeral storage
|
|
7
|
+
*/
|
|
8
|
+
export class MemoryTokenStore {
|
|
9
|
+
tokens = new Map();
|
|
10
|
+
async saveToken(key, token) {
|
|
11
|
+
this.tokens.set(key, token);
|
|
12
|
+
}
|
|
13
|
+
async getToken(key) {
|
|
14
|
+
const token = this.tokens.get(key);
|
|
15
|
+
if (!token)
|
|
16
|
+
return null;
|
|
17
|
+
// Check if expired
|
|
18
|
+
if (this.isTokenExpired(token)) {
|
|
19
|
+
this.tokens.delete(key);
|
|
20
|
+
return null;
|
|
21
|
+
}
|
|
22
|
+
return token;
|
|
23
|
+
}
|
|
24
|
+
async deleteToken(key) {
|
|
25
|
+
this.tokens.delete(key);
|
|
26
|
+
}
|
|
27
|
+
async listKeys() {
|
|
28
|
+
return Array.from(this.tokens.keys());
|
|
29
|
+
}
|
|
30
|
+
async clear() {
|
|
31
|
+
this.tokens.clear();
|
|
32
|
+
}
|
|
33
|
+
isTokenExpired(token) {
|
|
34
|
+
return Date.now() > token.expires_at;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* File-based token store with encryption
|
|
39
|
+
* For CLI applications and development
|
|
40
|
+
*/
|
|
41
|
+
export class FileTokenStore {
|
|
42
|
+
storePath;
|
|
43
|
+
encryptionKey;
|
|
44
|
+
constructor(storePath, encryptionKey) {
|
|
45
|
+
this.storePath = storePath;
|
|
46
|
+
this.encryptionKey = encryptionKey;
|
|
47
|
+
}
|
|
48
|
+
async saveToken(key, token) {
|
|
49
|
+
const tokens = await this.loadTokens();
|
|
50
|
+
tokens[key] = token;
|
|
51
|
+
await this.saveTokens(tokens);
|
|
52
|
+
}
|
|
53
|
+
async getToken(key) {
|
|
54
|
+
const tokens = await this.loadTokens();
|
|
55
|
+
const token = tokens[key];
|
|
56
|
+
if (!token)
|
|
57
|
+
return null;
|
|
58
|
+
// Check if expired
|
|
59
|
+
if (this.isTokenExpired(token)) {
|
|
60
|
+
delete tokens[key];
|
|
61
|
+
await this.saveTokens(tokens);
|
|
62
|
+
return null;
|
|
63
|
+
}
|
|
64
|
+
return token;
|
|
65
|
+
}
|
|
66
|
+
async deleteToken(key) {
|
|
67
|
+
const tokens = await this.loadTokens();
|
|
68
|
+
delete tokens[key];
|
|
69
|
+
await this.saveTokens(tokens);
|
|
70
|
+
}
|
|
71
|
+
async listKeys() {
|
|
72
|
+
const tokens = await this.loadTokens();
|
|
73
|
+
return Object.keys(tokens);
|
|
74
|
+
}
|
|
75
|
+
async clear() {
|
|
76
|
+
await this.saveTokens({});
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Load tokens from file
|
|
80
|
+
*/
|
|
81
|
+
async loadTokens() {
|
|
82
|
+
try {
|
|
83
|
+
// Ensure directory exists
|
|
84
|
+
await fs.mkdir(path.dirname(this.storePath), { recursive: true });
|
|
85
|
+
// Read file
|
|
86
|
+
const data = await fs.readFile(this.storePath, 'utf-8');
|
|
87
|
+
// Decrypt if encryption is enabled
|
|
88
|
+
const content = this.decrypt(data);
|
|
89
|
+
return JSON.parse(content);
|
|
90
|
+
}
|
|
91
|
+
catch (error) {
|
|
92
|
+
if (error instanceof Error && error.code === 'ENOENT') {
|
|
93
|
+
return {}; // File doesn't exist yet
|
|
94
|
+
}
|
|
95
|
+
throw error;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* Save tokens to file
|
|
100
|
+
*/
|
|
101
|
+
async saveTokens(tokens) {
|
|
102
|
+
const content = JSON.stringify(tokens, null, 2);
|
|
103
|
+
// Encrypt if encryption is enabled
|
|
104
|
+
const data = this.encrypt(content);
|
|
105
|
+
// Write to file with restricted permissions
|
|
106
|
+
await fs.writeFile(this.storePath, data, {
|
|
107
|
+
mode: 0o600, // Read/write for owner only
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Encrypt data using AES-256-GCM
|
|
112
|
+
*/
|
|
113
|
+
encrypt(data) {
|
|
114
|
+
if (!this.encryptionKey)
|
|
115
|
+
return data;
|
|
116
|
+
const key = this.deriveKey(this.encryptionKey);
|
|
117
|
+
const iv = crypto.randomBytes(16);
|
|
118
|
+
const cipher = crypto.createCipheriv('aes-256-gcm', key, iv);
|
|
119
|
+
let encrypted = cipher.update(data, 'utf8', 'hex');
|
|
120
|
+
encrypted += cipher.final('hex');
|
|
121
|
+
const authTag = cipher.getAuthTag();
|
|
122
|
+
// Return: iv:authTag:encrypted
|
|
123
|
+
return `${iv.toString('hex')}:${authTag.toString('hex')}:${encrypted}`;
|
|
124
|
+
}
|
|
125
|
+
/**
|
|
126
|
+
* Decrypt data using AES-256-GCM
|
|
127
|
+
*/
|
|
128
|
+
decrypt(data) {
|
|
129
|
+
if (!this.encryptionKey)
|
|
130
|
+
return data;
|
|
131
|
+
const key = this.deriveKey(this.encryptionKey);
|
|
132
|
+
const parts = data.split(':');
|
|
133
|
+
if (parts.length !== 3) {
|
|
134
|
+
throw new Error('Invalid encrypted data format');
|
|
135
|
+
}
|
|
136
|
+
const [ivHex, authTagHex, encrypted] = parts;
|
|
137
|
+
const iv = Buffer.from(ivHex, 'hex');
|
|
138
|
+
const authTag = Buffer.from(authTagHex, 'hex');
|
|
139
|
+
const decipher = crypto.createDecipheriv('aes-256-gcm', key, iv);
|
|
140
|
+
decipher.setAuthTag(authTag);
|
|
141
|
+
let decrypted = decipher.update(encrypted, 'hex', 'utf8');
|
|
142
|
+
decrypted += decipher.final('utf8');
|
|
143
|
+
return decrypted;
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Derive 256-bit key from password using PBKDF2
|
|
147
|
+
*/
|
|
148
|
+
deriveKey(password) {
|
|
149
|
+
// Use a fixed salt for key derivation
|
|
150
|
+
// In production, consider using a per-user salt
|
|
151
|
+
const salt = 'nitrostack-token-store';
|
|
152
|
+
return crypto.pbkdf2Sync(password, salt, 100000, 32, 'sha256');
|
|
153
|
+
}
|
|
154
|
+
isTokenExpired(token) {
|
|
155
|
+
return Date.now() > token.expires_at;
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
/**
|
|
159
|
+
* Create default token store
|
|
160
|
+
*
|
|
161
|
+
* Uses file-based storage with encryption if password provided
|
|
162
|
+
*
|
|
163
|
+
* @param storePath - Path to store tokens (default: ~/.nitrostack/tokens.json)
|
|
164
|
+
* @param encryptionKey - Optional encryption key
|
|
165
|
+
*/
|
|
166
|
+
export function createDefaultTokenStore(storePath, encryptionKey) {
|
|
167
|
+
const defaultPath = storePath || getDefaultStorePath();
|
|
168
|
+
if (encryptionKey) {
|
|
169
|
+
return new FileTokenStore(defaultPath, encryptionKey);
|
|
170
|
+
}
|
|
171
|
+
return new FileTokenStore(defaultPath);
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Get default token store path
|
|
175
|
+
*/
|
|
176
|
+
function getDefaultStorePath() {
|
|
177
|
+
const home = process.env.HOME || process.env.USERPROFILE || '';
|
|
178
|
+
return path.join(home, '.nitrostack', 'tokens.json');
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* Utility: Check if token is expired
|
|
182
|
+
*/
|
|
183
|
+
export function isTokenExpired(token) {
|
|
184
|
+
return Date.now() > token.expires_at;
|
|
185
|
+
}
|
|
186
|
+
/**
|
|
187
|
+
* Utility: Calculate token expiration timestamp
|
|
188
|
+
*/
|
|
189
|
+
export function calculateExpiration(expiresIn) {
|
|
190
|
+
return Date.now() + expiresIn * 1000;
|
|
191
|
+
}
|
|
192
|
+
/**
|
|
193
|
+
* Utility: Convert TokenResponse to StoredToken
|
|
194
|
+
*/
|
|
195
|
+
export function tokenResponseToStored(response, resource) {
|
|
196
|
+
return {
|
|
197
|
+
access_token: response.access_token,
|
|
198
|
+
token_type: response.token_type,
|
|
199
|
+
expires_at: response.expires_in ? calculateExpiration(response.expires_in) : Date.now() + 3600000, // Default 1 hour
|
|
200
|
+
refresh_token: response.refresh_token,
|
|
201
|
+
scope: response.scope,
|
|
202
|
+
resource,
|
|
203
|
+
};
|
|
204
|
+
}
|
|
205
|
+
//# sourceMappingURL=token-store.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"token-store.js","sourceRoot":"","sources":["../../src/auth/token-store.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { TokenIntrospection, McpAuthConfig } from './types.js';
|
|
2
|
+
interface TokenValidationResult {
|
|
3
|
+
valid: boolean;
|
|
4
|
+
introspection?: TokenIntrospection;
|
|
5
|
+
error?: string;
|
|
6
|
+
}
|
|
7
|
+
/**
|
|
8
|
+
* Validate a Bearer token
|
|
9
|
+
*
|
|
10
|
+
* @param token - The access token to validate
|
|
11
|
+
* @param config - Auth configuration
|
|
12
|
+
* @returns Validation result with token introspection data
|
|
13
|
+
*/
|
|
14
|
+
export declare function validateToken(token: string, config: McpAuthConfig): Promise<TokenValidationResult>;
|
|
15
|
+
/**
|
|
16
|
+
* Introspect token using OAuth 2.0 Token Introspection (RFC 7662)
|
|
17
|
+
*
|
|
18
|
+
* @internal - Exported for testing
|
|
19
|
+
*/
|
|
20
|
+
export declare function introspectToken(token: string, config: McpAuthConfig): Promise<TokenIntrospection>;
|
|
21
|
+
/**
|
|
22
|
+
* Validate JWT using JWKS
|
|
23
|
+
*
|
|
24
|
+
* @internal - Exported for testing
|
|
25
|
+
*/
|
|
26
|
+
export declare function validateJWT(token: string, config: McpAuthConfig): Promise<TokenIntrospection>;
|
|
27
|
+
/**
|
|
28
|
+
* Validate token audience (RFC 8707)
|
|
29
|
+
*
|
|
30
|
+
* CRITICAL: This prevents confused deputy attacks
|
|
31
|
+
* Token MUST be issued specifically for this resource
|
|
32
|
+
*
|
|
33
|
+
* @param introspection - Token introspection result
|
|
34
|
+
* @param expectedAudience - Expected audience value(s)
|
|
35
|
+
* @returns true if audience is valid
|
|
36
|
+
*/
|
|
37
|
+
export declare function validateAudience(introspection: TokenIntrospection, expectedAudience?: string | string[]): boolean;
|
|
38
|
+
/**
|
|
39
|
+
* Validate scopes
|
|
40
|
+
*
|
|
41
|
+
* @param introspection - Token introspection result
|
|
42
|
+
* @param requiredScopes - Scopes required for the operation
|
|
43
|
+
* @returns true if token has all required scopes
|
|
44
|
+
*/
|
|
45
|
+
export declare function validateScopes(introspection: TokenIntrospection, requiredScopes: string[]): boolean;
|
|
46
|
+
/**
|
|
47
|
+
* Extract Bearer token from Authorization header
|
|
48
|
+
*/
|
|
49
|
+
export declare function extractBearerToken(authHeader: string | undefined): string | null;
|
|
50
|
+
/**
|
|
51
|
+
* Check if token is expired
|
|
52
|
+
*/
|
|
53
|
+
export declare function isTokenExpired(introspection: TokenIntrospection): boolean;
|
|
54
|
+
/**
|
|
55
|
+
* Clear token cache (useful for testing)
|
|
56
|
+
*/
|
|
57
|
+
export declare function clearTokenCache(): void;
|
|
58
|
+
export {};
|
|
59
|
+
//# sourceMappingURL=token-validation.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"token-validation.d.ts","sourceRoot":"","sources":["../../src/auth/token-validation.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,aAAa,EAAE,MAAM,YAAY,CAAC;AAqB/D,UAAU,qBAAqB;IAC7B,KAAK,EAAE,OAAO,CAAC;IACf,aAAa,CAAC,EAAE,kBAAkB,CAAC;IACnC,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAQD;;;;;;GAMG;AACH,wBAAsB,aAAa,CACjC,KAAK,EAAE,MAAM,EACb,MAAM,EAAE,aAAa,GACpB,OAAO,CAAC,qBAAqB,CAAC,CA0ChC;AAED;;;;GAIG;AACH,wBAAsB,eAAe,CACnC,KAAK,EAAE,MAAM,EACb,MAAM,EAAE,aAAa,GACpB,OAAO,CAAC,kBAAkB,CAAC,CAmC7B;AAED;;;;GAIG;AACH,wBAAsB,WAAW,CAC/B,KAAK,EAAE,MAAM,EACb,MAAM,EAAE,aAAa,GACpB,OAAO,CAAC,kBAAkB,CAAC,CAgC7B;AAED;;;;;;;;;GASG;AACH,wBAAgB,gBAAgB,CAC9B,aAAa,EAAE,kBAAkB,EACjC,gBAAgB,CAAC,EAAE,MAAM,GAAG,MAAM,EAAE,GACnC,OAAO,CAgBT;AAED;;;;;;GAMG;AACH,wBAAgB,cAAc,CAC5B,aAAa,EAAE,kBAAkB,EACjC,cAAc,EAAE,MAAM,EAAE,GACvB,OAAO,CAWT;AAED;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,UAAU,EAAE,MAAM,GAAG,SAAS,GAAG,MAAM,GAAG,IAAI,CAOhF;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,aAAa,EAAE,kBAAkB,GAAG,OAAO,CAOzE;AA4CD;;GAEG;AACH,wBAAgB,eAAe,IAAI,IAAI,CAEtC"}
|