@backstage/plugin-auth-backend-module-pinniped-provider 0.1.1-next.1 → 0.1.1

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/CHANGELOG.md CHANGED
@@ -1,5 +1,25 @@
1
1
  # @backstage/plugin-auth-backend-module-pinniped-provider
2
2
 
3
+ ## 0.1.1
4
+
5
+ ### Patch Changes
6
+
7
+ - a8f6afda4a: Introduced metadata cache for the `pinniped` provider.
8
+ - Updated dependencies
9
+ - @backstage/backend-common@0.19.9
10
+ - @backstage/backend-plugin-api@0.6.7
11
+ - @backstage/config@1.1.1
12
+ - @backstage/plugin-auth-node@0.4.1
13
+
14
+ ## 0.1.1-next.2
15
+
16
+ ### Patch Changes
17
+
18
+ - Updated dependencies
19
+ - @backstage/backend-plugin-api@0.6.7-next.2
20
+ - @backstage/backend-common@0.19.9-next.2
21
+ - @backstage/plugin-auth-node@0.4.1-next.2
22
+
3
23
  ## 0.1.1-next.1
4
24
 
5
25
  ### Patch Changes
package/dist/index.cjs.js CHANGED
@@ -4,8 +4,15 @@ Object.defineProperty(exports, '__esModule', { value: true });
4
4
 
5
5
  var pluginAuthNode = require('@backstage/plugin-auth-node');
6
6
  var openidClient = require('openid-client');
7
+ var luxon = require('luxon');
7
8
  var backendPluginApi = require('@backstage/backend-plugin-api');
8
9
 
10
+ var __defProp = Object.defineProperty;
11
+ var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
12
+ var __publicField = (obj, key, value) => {
13
+ __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
14
+ return value;
15
+ };
9
16
  const rfc8693TokenExchange = async ({
10
17
  subject_token,
11
18
  target_audience,
@@ -22,22 +29,50 @@ const rfc8693TokenExchange = async ({
22
29
  throw new Error(`RFC8693 token exchange failed with error: ${err}`);
23
30
  });
24
31
  };
25
- const pinnipedAuthenticator = pluginAuthNode.createOAuthAuthenticator({
26
- defaultProfileTransform: async (_r, _c) => ({ profile: {} }),
27
- async initialize({ callbackUrl, config }) {
32
+ const OIDC_METADATA_TTL_SECONDS = 3600;
33
+ class PinnipedStrategyCache {
34
+ constructor(callbackUrl, config) {
35
+ __publicField(this, "callbackUrl");
36
+ __publicField(this, "config");
37
+ __publicField(this, "strategyPromise");
38
+ __publicField(this, "cachedPromise");
39
+ __publicField(this, "cachedPromiseExpiry");
40
+ this.callbackUrl = callbackUrl;
41
+ this.config = config;
42
+ this.strategyPromise = this.buildStrategy();
43
+ }
44
+ async getStrategy() {
45
+ if (this.cachedPromise) {
46
+ if (this.cachedPromiseExpiry && luxon.DateTime.fromJSDate(this.cachedPromiseExpiry) > luxon.DateTime.local()) {
47
+ return this.cachedPromise;
48
+ }
49
+ this.strategyPromise = this.buildStrategy();
50
+ delete this.cachedPromise;
51
+ }
52
+ try {
53
+ await this.strategyPromise;
54
+ this.cachedPromise = this.strategyPromise;
55
+ this.cachedPromiseExpiry = luxon.DateTime.utc().plus({ seconds: OIDC_METADATA_TTL_SECONDS }).toJSDate();
56
+ } catch (error) {
57
+ this.strategyPromise = this.buildStrategy();
58
+ delete this.cachedPromise;
59
+ delete this.cachedPromiseExpiry;
60
+ }
61
+ return this.strategyPromise;
62
+ }
63
+ async buildStrategy() {
28
64
  const issuer = await openidClient.Issuer.discover(
29
- `${config.getString(
65
+ `${this.config.getString(
30
66
  "federationDomain"
31
67
  )}/.well-known/openid-configuration`
32
68
  );
33
69
  const client = new issuer.Client({
34
70
  access_type: "offline",
35
- // this option must be passed to provider to receive a refresh token
36
- client_id: config.getString("clientId"),
37
- client_secret: config.getString("clientSecret"),
38
- redirect_uris: [callbackUrl],
71
+ client_id: this.config.getString("clientId"),
72
+ client_secret: this.config.getString("clientSecret"),
73
+ redirect_uris: [this.callbackUrl],
39
74
  response_types: ["code"],
40
- scope: config.getOptionalString("scope") || "",
75
+ scope: this.config.getOptionalString("scope") || "",
41
76
  id_token_signed_response_alg: "ES256"
42
77
  });
43
78
  const providerStrategy = new openidClient.Strategy(
@@ -50,10 +85,16 @@ const pinnipedAuthenticator = pluginAuthNode.createOAuthAuthenticator({
50
85
  }
51
86
  );
52
87
  return { providerStrategy, client };
88
+ }
89
+ }
90
+ const pinnipedAuthenticator = pluginAuthNode.createOAuthAuthenticator({
91
+ defaultProfileTransform: async (_r, _c) => ({ profile: {} }),
92
+ initialize({ callbackUrl, config }) {
93
+ return new PinnipedStrategyCache(callbackUrl, config);
53
94
  },
54
95
  async start(input, ctx) {
55
96
  var _a;
56
- const { providerStrategy } = await ctx;
97
+ const { providerStrategy } = await ctx.getStrategy();
57
98
  const stringifiedAudience = (_a = input.req.query) == null ? void 0 : _a.audience;
58
99
  const decodedState = pluginAuthNode.decodeOAuthState(input.state);
59
100
  const state = { ...decodedState, audience: stringifiedAudience };
@@ -73,7 +114,7 @@ const pinnipedAuthenticator = pluginAuthNode.createOAuthAuthenticator({
73
114
  });
74
115
  },
75
116
  async authenticate(input, ctx) {
76
- const { providerStrategy } = await ctx;
117
+ const { providerStrategy } = await ctx.getStrategy();
77
118
  const { req } = input;
78
119
  const { searchParams } = new URL(req.url, "https://pinniped.com");
79
120
  const stateParam = searchParams.get("state");
@@ -84,7 +125,7 @@ const pinnipedAuthenticator = pluginAuthNode.createOAuthAuthenticator({
84
125
  (audience ? rfc8693TokenExchange({
85
126
  subject_token: user.tokenset.access_token,
86
127
  target_audience: audience,
87
- ctx
128
+ ctx: ctx.getStrategy()
88
129
  }).catch(
89
130
  (err) => reject(
90
131
  new Error(
@@ -118,7 +159,7 @@ const pinnipedAuthenticator = pluginAuthNode.createOAuthAuthenticator({
118
159
  });
119
160
  },
120
161
  async refresh(input, ctx) {
121
- const { client } = await ctx;
162
+ const { client } = await ctx.getStrategy();
122
163
  const tokenset = await client.refresh(input.refreshToken);
123
164
  return new Promise((resolve, reject) => {
124
165
  var _a;
@@ -162,6 +203,7 @@ const authModulePinnipedProvider = backendPluginApi.createBackendModule({
162
203
  }
163
204
  });
164
205
 
206
+ exports.PinnipedStrategyCache = PinnipedStrategyCache;
165
207
  exports.authModulePinnipedProvider = authModulePinnipedProvider;
166
208
  exports.pinnipedAuthenticator = pinnipedAuthenticator;
167
209
  //# sourceMappingURL=index.cjs.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs.js","sources":["../src/authenticator.ts","../src/module.ts"],"sourcesContent":["/*\n * Copyright 2023 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport { PassportDoneCallback } from '@backstage/plugin-auth-node';\nimport {\n createOAuthAuthenticator,\n decodeOAuthState,\n encodeOAuthState,\n} from '@backstage/plugin-auth-node';\nimport {\n Client,\n Issuer,\n TokenSet,\n Strategy as OidcStrategy,\n} from 'openid-client';\n\nconst rfc8693TokenExchange = async ({\n subject_token,\n target_audience,\n ctx,\n}: {\n subject_token: string;\n target_audience: string;\n ctx: Promise<{\n providerStrategy: OidcStrategy<{}>;\n client: Client;\n }>;\n}): Promise<string | undefined> => {\n const { client } = await ctx;\n return client\n .grant({\n grant_type: 'urn:ietf:params:oauth:grant-type:token-exchange',\n subject_token,\n audience: target_audience,\n subject_token_type: 'urn:ietf:params:oauth:token-type:access_token',\n requested_token_type: 'urn:ietf:params:oauth:token-type:jwt',\n })\n .then(tokenset => tokenset.access_token)\n .catch(err => {\n throw new Error(`RFC8693 token exchange failed with error: ${err}`);\n });\n};\n\n/** @public */\nexport const pinnipedAuthenticator = createOAuthAuthenticator({\n defaultProfileTransform: async (_r, _c) => ({ profile: {} }),\n async initialize({ callbackUrl, config }) {\n const issuer = await Issuer.discover(\n `${config.getString(\n 'federationDomain',\n )}/.well-known/openid-configuration`,\n );\n const client = new issuer.Client({\n access_type: 'offline', // this option must be passed to provider to receive a refresh token\n client_id: config.getString('clientId'),\n client_secret: config.getString('clientSecret'),\n redirect_uris: [callbackUrl],\n response_types: ['code'],\n scope: config.getOptionalString('scope') || '',\n id_token_signed_response_alg: 'ES256',\n });\n const providerStrategy = new OidcStrategy(\n {\n client,\n passReqToCallback: false,\n },\n (\n tokenset: TokenSet,\n done: PassportDoneCallback<\n { tokenset: TokenSet },\n {\n refreshToken?: string;\n }\n >,\n ) => {\n done(undefined, { tokenset }, {});\n },\n );\n\n return { providerStrategy, client };\n },\n\n async start(input, ctx) {\n const { providerStrategy } = await ctx;\n const stringifiedAudience = input.req.query?.audience as string;\n const decodedState = decodeOAuthState(input.state);\n const state = { ...decodedState, audience: stringifiedAudience };\n const options: Record<string, string> = {\n scope:\n input.scope ||\n 'openid pinniped:request-audience username offline_access',\n state: encodeOAuthState(state),\n };\n\n return new Promise((resolve, reject) => {\n const strategy = Object.create(providerStrategy);\n strategy.redirect = (url: string) => {\n resolve({ url });\n };\n strategy.error = (error: Error) => {\n reject(error);\n };\n strategy.authenticate(input.req, { ...options });\n });\n },\n\n async authenticate(input, ctx) {\n const { providerStrategy } = await ctx;\n const { req } = input;\n const { searchParams } = new URL(req.url, 'https://pinniped.com');\n const stateParam = searchParams.get('state');\n const audience = stateParam\n ? decodeOAuthState(stateParam).audience\n : undefined;\n\n return new Promise((resolve, reject) => {\n const strategy = Object.create(providerStrategy);\n strategy.success = (user: any) => {\n (audience\n ? rfc8693TokenExchange({\n subject_token: user.tokenset.access_token,\n target_audience: audience,\n ctx,\n }).catch(err =>\n reject(\n new Error(\n `Failed to get cluster specific ID token for \"${audience}\": ${err}`,\n ),\n ),\n )\n : Promise.resolve(user.tokenset.id_token)\n ).then(idToken => {\n resolve({\n fullProfile: { provider: '', id: '', displayName: '' },\n session: {\n accessToken: user.tokenset.access_token!,\n tokenType: user.tokenset.token_type ?? 'bearer',\n scope: user.tokenset.scope!,\n idToken,\n refreshToken: user.tokenset.refresh_token,\n },\n });\n });\n };\n\n strategy.fail = (info: any) => {\n reject(new Error(`Authentication rejected, ${info.message || ''}`));\n };\n\n strategy.error = (error: Error) => {\n reject(error);\n };\n\n strategy.redirect = () => {\n reject(new Error('Unexpected redirect'));\n };\n\n strategy.authenticate(req);\n });\n },\n\n async refresh(input, ctx) {\n const { client } = await ctx;\n const tokenset = await client.refresh(input.refreshToken);\n\n return new Promise((resolve, reject) => {\n if (!tokenset.access_token) {\n reject(new Error('Refresh Failed'));\n }\n\n resolve({\n fullProfile: { provider: '', id: '', displayName: '' },\n session: {\n accessToken: tokenset.access_token!,\n tokenType: tokenset.token_type ?? 'bearer',\n scope: tokenset.scope!,\n idToken: tokenset.id_token,\n refreshToken: tokenset.refresh_token,\n },\n });\n });\n },\n});\n","/*\n * Copyright 2023 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport { createBackendModule } from '@backstage/backend-plugin-api';\nimport {\n authProvidersExtensionPoint,\n commonSignInResolvers,\n createOAuthProviderFactory,\n} from '@backstage/plugin-auth-node';\nimport { pinnipedAuthenticator } from './authenticator';\n\n/** @public */\nexport const authModulePinnipedProvider = createBackendModule({\n pluginId: 'auth',\n moduleId: 'pinniped-provider',\n register(reg) {\n reg.registerInit({\n deps: {\n providers: authProvidersExtensionPoint,\n },\n async init({ providers }) {\n providers.registerProvider({\n providerId: 'pinniped',\n factory: createOAuthProviderFactory({\n authenticator: pinnipedAuthenticator,\n signInResolverFactories: {\n ...commonSignInResolvers,\n },\n }),\n });\n },\n });\n },\n});\n"],"names":["createOAuthAuthenticator","Issuer","OidcStrategy","decodeOAuthState","encodeOAuthState","createBackendModule","authProvidersExtensionPoint","createOAuthProviderFactory","commonSignInResolvers"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs.js","sources":["../src/authenticator.ts","../src/module.ts"],"sourcesContent":["/*\n * Copyright 2023 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport { Config } from '@backstage/config';\nimport { PassportDoneCallback } from '@backstage/plugin-auth-node';\nimport {\n createOAuthAuthenticator,\n decodeOAuthState,\n encodeOAuthState,\n} from '@backstage/plugin-auth-node';\nimport {\n Client,\n Issuer,\n TokenSet,\n Strategy as OidcStrategy,\n BaseClient,\n} from 'openid-client';\nimport { DateTime } from 'luxon';\n\nconst rfc8693TokenExchange = async ({\n subject_token,\n target_audience,\n ctx,\n}: {\n subject_token: string;\n target_audience: string;\n ctx: Promise<{\n providerStrategy: OidcStrategy<{}>;\n client: Client;\n }>;\n}): Promise<string | undefined> => {\n const { client } = await ctx;\n return client\n .grant({\n grant_type: 'urn:ietf:params:oauth:grant-type:token-exchange',\n subject_token,\n audience: target_audience,\n subject_token_type: 'urn:ietf:params:oauth:token-type:access_token',\n requested_token_type: 'urn:ietf:params:oauth:token-type:jwt',\n })\n .then(tokenset => tokenset.access_token)\n .catch(err => {\n throw new Error(`RFC8693 token exchange failed with error: ${err}`);\n });\n};\n\nconst OIDC_METADATA_TTL_SECONDS = 3600;\n\n/** @public */\nexport class PinnipedStrategyCache {\n private readonly callbackUrl: string;\n private readonly config: Config;\n private strategyPromise: Promise<{\n providerStrategy: OidcStrategy<{ tokenset: TokenSet }, BaseClient>;\n client: BaseClient;\n }>;\n\n private cachedPromise?: Promise<{\n providerStrategy: OidcStrategy<{ tokenset: TokenSet }, BaseClient>;\n client: BaseClient;\n }>;\n private cachedPromiseExpiry?: Date;\n\n constructor(callbackUrl: string, config: Config) {\n this.callbackUrl = callbackUrl;\n this.config = config;\n this.strategyPromise = this.buildStrategy();\n }\n\n public async getStrategy(): Promise<{\n providerStrategy: OidcStrategy<{ tokenset: TokenSet }, BaseClient>;\n client: BaseClient;\n }> {\n if (this.cachedPromise) {\n if (\n this.cachedPromiseExpiry &&\n DateTime.fromJSDate(this.cachedPromiseExpiry) > DateTime.local()\n ) {\n return this.cachedPromise;\n }\n // cachedPromise has expired, remove promise from cache and regenerate strategy\n this.strategyPromise = this.buildStrategy();\n delete this.cachedPromise;\n }\n\n try {\n // if strategy is generated successfully, save it to cache\n await this.strategyPromise;\n this.cachedPromise = this.strategyPromise;\n this.cachedPromiseExpiry = DateTime.utc()\n .plus({ seconds: OIDC_METADATA_TTL_SECONDS })\n .toJSDate();\n } catch (error) {\n // if we fail to generate a strategy, retry and overwrite strategy\n this.strategyPromise = this.buildStrategy();\n delete this.cachedPromise;\n delete this.cachedPromiseExpiry;\n }\n\n return this.strategyPromise;\n }\n\n private async buildStrategy(): Promise<{\n providerStrategy: OidcStrategy<{ tokenset: TokenSet }, BaseClient>;\n client: BaseClient;\n }> {\n const issuer = await Issuer.discover(\n `${this.config.getString(\n 'federationDomain',\n )}/.well-known/openid-configuration`,\n );\n const client = new issuer.Client({\n access_type: 'offline',\n client_id: this.config.getString('clientId'),\n client_secret: this.config.getString('clientSecret'),\n redirect_uris: [this.callbackUrl],\n response_types: ['code'],\n scope: this.config.getOptionalString('scope') || '',\n id_token_signed_response_alg: 'ES256',\n });\n const providerStrategy = new OidcStrategy(\n {\n client,\n passReqToCallback: false,\n },\n (\n tokenset: TokenSet,\n done: PassportDoneCallback<\n { tokenset: TokenSet },\n {\n refreshToken?: string;\n }\n >,\n ) => {\n done(undefined, { tokenset }, {});\n },\n );\n return { providerStrategy, client };\n }\n}\n\n/** @public */\nexport const pinnipedAuthenticator = createOAuthAuthenticator({\n defaultProfileTransform: async (_r, _c) => ({ profile: {} }),\n initialize({ callbackUrl, config }) {\n return new PinnipedStrategyCache(callbackUrl, config);\n },\n async start(input, ctx): Promise<{ url: string; status?: number }> {\n const { providerStrategy } = await ctx.getStrategy();\n const stringifiedAudience = input.req.query?.audience as string;\n const decodedState = decodeOAuthState(input.state);\n const state = { ...decodedState, audience: stringifiedAudience };\n const options: Record<string, string> = {\n scope:\n input.scope ||\n 'openid pinniped:request-audience username offline_access',\n state: encodeOAuthState(state),\n };\n\n return new Promise((resolve, reject) => {\n const strategy = Object.create(providerStrategy);\n strategy.redirect = (url: string) => {\n resolve({ url });\n };\n strategy.error = (error: Error) => {\n reject(error);\n };\n strategy.authenticate(input.req, { ...options });\n });\n },\n\n async authenticate(input, ctx) {\n const { providerStrategy } = await ctx.getStrategy();\n const { req } = input;\n const { searchParams } = new URL(req.url, 'https://pinniped.com');\n const stateParam = searchParams.get('state');\n const audience = stateParam\n ? decodeOAuthState(stateParam).audience\n : undefined;\n\n return new Promise((resolve, reject) => {\n const strategy = Object.create(providerStrategy);\n strategy.success = (user: any) => {\n (audience\n ? rfc8693TokenExchange({\n subject_token: user.tokenset.access_token,\n target_audience: audience,\n ctx: ctx.getStrategy(),\n }).catch(err =>\n reject(\n new Error(\n `Failed to get cluster specific ID token for \"${audience}\": ${err}`,\n ),\n ),\n )\n : Promise.resolve(user.tokenset.id_token)\n ).then(idToken => {\n resolve({\n fullProfile: { provider: '', id: '', displayName: '' },\n session: {\n accessToken: user.tokenset.access_token!,\n tokenType: user.tokenset.token_type ?? 'bearer',\n scope: user.tokenset.scope!,\n idToken,\n refreshToken: user.tokenset.refresh_token,\n },\n });\n });\n };\n\n strategy.fail = (info: any) => {\n reject(new Error(`Authentication rejected, ${info.message || ''}`));\n };\n\n strategy.error = (error: Error) => {\n reject(error);\n };\n\n strategy.redirect = () => {\n reject(new Error('Unexpected redirect'));\n };\n\n strategy.authenticate(req);\n });\n },\n\n async refresh(input, ctx) {\n const { client } = await ctx.getStrategy();\n const tokenset = await client.refresh(input.refreshToken);\n\n return new Promise((resolve, reject) => {\n if (!tokenset.access_token) {\n reject(new Error('Refresh Failed'));\n }\n\n resolve({\n fullProfile: { provider: '', id: '', displayName: '' },\n session: {\n accessToken: tokenset.access_token!,\n tokenType: tokenset.token_type ?? 'bearer',\n scope: tokenset.scope!,\n idToken: tokenset.id_token,\n refreshToken: tokenset.refresh_token,\n },\n });\n });\n },\n});\n","/*\n * Copyright 2023 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport { createBackendModule } from '@backstage/backend-plugin-api';\nimport {\n authProvidersExtensionPoint,\n commonSignInResolvers,\n createOAuthProviderFactory,\n} from '@backstage/plugin-auth-node';\nimport { pinnipedAuthenticator } from './authenticator';\n\n/** @public */\nexport const authModulePinnipedProvider = createBackendModule({\n pluginId: 'auth',\n moduleId: 'pinniped-provider',\n register(reg) {\n reg.registerInit({\n deps: {\n providers: authProvidersExtensionPoint,\n },\n async init({ providers }) {\n providers.registerProvider({\n providerId: 'pinniped',\n factory: createOAuthProviderFactory({\n authenticator: pinnipedAuthenticator,\n signInResolverFactories: {\n ...commonSignInResolvers,\n },\n }),\n });\n },\n });\n },\n});\n"],"names":["DateTime","Issuer","OidcStrategy","createOAuthAuthenticator","decodeOAuthState","encodeOAuthState","createBackendModule","authProvidersExtensionPoint","createOAuthProviderFactory","commonSignInResolvers"],"mappings":"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"}
package/dist/index.d.ts CHANGED
@@ -1,17 +1,28 @@
1
1
  import * as _backstage_plugin_auth_node from '@backstage/plugin-auth-node';
2
- import * as openid_client from 'openid-client';
3
- import { Strategy, TokenSet } from 'openid-client';
2
+ import { Config } from '@backstage/config';
3
+ import { Strategy, TokenSet, BaseClient } from 'openid-client';
4
4
  import * as _backstage_backend_plugin_api from '@backstage/backend-plugin-api';
5
5
 
6
6
  /** @public */
7
- declare const pinnipedAuthenticator: _backstage_plugin_auth_node.OAuthAuthenticator<Promise<{
8
- providerStrategy: Strategy<{
9
- tokenset: TokenSet;
10
- }, openid_client.BaseClient>;
11
- client: openid_client.BaseClient;
12
- }>, unknown>;
7
+ declare class PinnipedStrategyCache {
8
+ private readonly callbackUrl;
9
+ private readonly config;
10
+ private strategyPromise;
11
+ private cachedPromise?;
12
+ private cachedPromiseExpiry?;
13
+ constructor(callbackUrl: string, config: Config);
14
+ getStrategy(): Promise<{
15
+ providerStrategy: Strategy<{
16
+ tokenset: TokenSet;
17
+ }, BaseClient>;
18
+ client: BaseClient;
19
+ }>;
20
+ private buildStrategy;
21
+ }
22
+ /** @public */
23
+ declare const pinnipedAuthenticator: _backstage_plugin_auth_node.OAuthAuthenticator<PinnipedStrategyCache, unknown>;
13
24
 
14
25
  /** @public */
15
26
  declare const authModulePinnipedProvider: () => _backstage_backend_plugin_api.BackendFeature;
16
27
 
17
- export { authModulePinnipedProvider, pinnipedAuthenticator };
28
+ export { PinnipedStrategyCache, authModulePinnipedProvider, pinnipedAuthenticator };
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@backstage/plugin-auth-backend-module-pinniped-provider",
3
3
  "description": "The pinniped-provider backend module for the auth plugin.",
4
- "version": "0.1.1-next.1",
4
+ "version": "0.1.1",
5
5
  "main": "dist/index.cjs.js",
6
6
  "types": "dist/index.d.ts",
7
7
  "license": "Apache-2.0",
@@ -23,17 +23,18 @@
23
23
  "postpack": "backstage-cli package postpack"
24
24
  },
25
25
  "dependencies": {
26
- "@backstage/backend-common": "^0.19.9-next.1",
27
- "@backstage/backend-plugin-api": "^0.6.7-next.1",
28
- "@backstage/plugin-auth-node": "^0.4.1-next.1",
26
+ "@backstage/backend-common": "^0.19.9",
27
+ "@backstage/backend-plugin-api": "^0.6.7",
28
+ "@backstage/config": "^1.1.1",
29
+ "@backstage/plugin-auth-node": "^0.4.1",
30
+ "luxon": "^3.4.3",
29
31
  "openid-client": "^5.4.3"
30
32
  },
31
33
  "devDependencies": {
32
- "@backstage/backend-defaults": "^0.2.7-next.1",
33
- "@backstage/backend-test-utils": "^0.2.8-next.1",
34
- "@backstage/cli": "^0.24.0-next.1",
35
- "@backstage/config": "^1.1.1",
36
- "@backstage/plugin-auth-backend": "^0.20.0-next.1",
34
+ "@backstage/backend-defaults": "^0.2.7",
35
+ "@backstage/backend-test-utils": "^0.2.8",
36
+ "@backstage/cli": "^0.24.0",
37
+ "@backstage/plugin-auth-backend": "^0.20.0",
37
38
  "cookie-parser": "^1.4.6",
38
39
  "express": "^4.18.2",
39
40
  "express-promise-router": "^4.1.1",