mirror of
https://github.com/hashicorp/vault.git
synced 2025-08-31 03:21:11 +02:00
* replace all injects with import syntax * Delete ui/app/components/identity/_popup-base.js
67 lines
2.3 KiB
JavaScript
67 lines
2.3 KiB
JavaScript
/**
|
|
* Copyright (c) HashiCorp, Inc.
|
|
* SPDX-License-Identifier: BUSL-1.1
|
|
*/
|
|
|
|
import ApplicationAdapter from './application';
|
|
import { service } from '@ember/service';
|
|
import { encodePath } from 'vault/utils/path-encoding-helpers';
|
|
import { v4 as uuidv4 } from 'uuid';
|
|
|
|
export default ApplicationAdapter.extend({
|
|
router: service(),
|
|
|
|
// generateClientChallenge generates a client challenge from a verifier.
|
|
// The client challenge is the base64(sha256(verifier)). The verifier is
|
|
// later presented to the server to obtain the resulting Vault token.
|
|
async generateClientChallenge(verifier) {
|
|
const encoder = new TextEncoder();
|
|
const data = encoder.encode(verifier);
|
|
const hashBuffer = await crypto.subtle.digest('SHA-256', data);
|
|
const hashArray = new Uint8Array(hashBuffer);
|
|
return btoa(String.fromCharCode.apply(null, hashArray));
|
|
},
|
|
|
|
async findRecord(store, type, id, snapshot) {
|
|
let [path, role] = JSON.parse(id);
|
|
path = preparePathSegment(path);
|
|
|
|
// Create the ACS URL based on the cluster the UI is targeting
|
|
let acs_url = `${window.location.origin}/v1/`;
|
|
let namespace = snapshot?.adapterOptions.namespace;
|
|
if (namespace) {
|
|
namespace = preparePathSegment(namespace);
|
|
acs_url = acs_url.concat(namespace, '/');
|
|
}
|
|
acs_url = acs_url.concat('auth/', path, '/callback');
|
|
|
|
// Create the client verifier and challenge
|
|
const verifier = uuidv4();
|
|
const challenge = await this.generateClientChallenge(verifier);
|
|
// Kick off the authentication flow by generating the SSO service URL
|
|
// It requires the client challenge generated from the verifier. We'll
|
|
// later provide the verifier to match up with the challenge on the server
|
|
// when we poll for the Vault token by its returned token poll ID.
|
|
const response = await this.ajax(`/v1/auth/${path}/sso_service_url`, 'PUT', {
|
|
data: {
|
|
acs_url,
|
|
role,
|
|
client_challenge: challenge,
|
|
client_type: 'browser',
|
|
},
|
|
});
|
|
return {
|
|
...response.data,
|
|
client_verifier: verifier,
|
|
};
|
|
},
|
|
});
|
|
|
|
// preparePathSegment prepares the given segment for being included in a URL
|
|
// path by trimming leading and trailing forward slashes and URL encoding.
|
|
function preparePathSegment(segment) {
|
|
segment = segment.replace(/^\//, '');
|
|
segment = segment.replace(/\/$/, '');
|
|
return encodePath(segment);
|
|
}
|