-
Notifications
You must be signed in to change notification settings - Fork 178
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add s3 storage artifact route and ui integration of it
- Loading branch information
1 parent
b33db83
commit 04a64fd
Showing
24 changed files
with
719 additions
and
36 deletions.
There are no files selected for viewing
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,63 @@ | ||
import { FastifyInstance, FastifyReply } from 'fastify'; | ||
import { createMinioClient, getObjectStream } from './storageUtils'; | ||
import { getDashboardConfig } from '../../../utils/resourceUtils'; | ||
import { getDirectCallOptions } from '../../../utils/directCallUtils'; | ||
import { getAccessToken } from '../../../utils/directCallUtils'; | ||
import { OauthFastifyRequest } from '../../../types'; | ||
|
||
export default async (fastify: FastifyInstance): Promise<void> => { | ||
const dashConfig = getDashboardConfig(); | ||
if (dashConfig?.spec.dashboardConfig.disableS3Endpoint === false) { | ||
fastify.get( | ||
'/:namespace/:bucket', | ||
async ( | ||
request: OauthFastifyRequest<{ | ||
Querystring: Record<string, string>; | ||
Params: { '*': string; [key: string]: string }; | ||
Body: { [key: string]: unknown }; | ||
}>, | ||
reply: FastifyReply, | ||
) => { | ||
try { | ||
const { namespace, bucket } = request.params; | ||
const query = request.query; | ||
const key = query.key; | ||
|
||
const requestOptions = await getDirectCallOptions(fastify, request, request.url); | ||
const token = getAccessToken(requestOptions); | ||
|
||
const stream = await getObjectStream({ | ||
bucket, | ||
client: await createMinioClient(fastify, token, namespace), | ||
key, | ||
}); | ||
|
||
reply.type('text/plain'); | ||
|
||
await new Promise<void>((resolve, reject) => { | ||
stream.on('data', (chunk) => { | ||
reply.raw.write(chunk); | ||
}); | ||
|
||
stream.on('end', () => { | ||
reply.raw.end(); | ||
resolve(); | ||
}); | ||
|
||
stream.on('error', (err) => { | ||
fastify.log.error('Stream error:', err); | ||
reply.raw.statusCode = 500; | ||
reply.raw.end(err.message); | ||
reject(err); | ||
}); | ||
}); | ||
|
||
return; | ||
} catch (err) { | ||
reply.code(500).send(err.message); | ||
return reply; | ||
} | ||
}, | ||
); | ||
} | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,118 @@ | ||
import { Client as MinioClient } from 'minio'; | ||
import { DSPipelineKind, KubeFastifyInstance } from '../../../types'; | ||
import { Transform, PassThrough } from 'stream'; | ||
|
||
/** | ||
* Create minio client with aws instance profile credentials if needed. | ||
* @param config minio client options where `accessKey` and `secretKey` are optional. | ||
*/ | ||
export async function createMinioClient( | ||
fastify: KubeFastifyInstance, | ||
token: string, | ||
namespace: string, | ||
): Promise<MinioClient> { | ||
try { | ||
const dspaResponse = await fastify.kube.customObjectsApi | ||
.listNamespacedCustomObject( | ||
'datasciencepipelinesapplications.opendatahub.io', | ||
'v1alpha1', | ||
namespace, | ||
'datasciencepipelinesapplications', | ||
undefined, | ||
undefined, | ||
undefined, | ||
undefined, | ||
undefined, | ||
undefined, | ||
undefined, | ||
undefined, | ||
{ | ||
headers: { | ||
authorization: `Bearer ${token}`, | ||
}, | ||
}, | ||
) | ||
.catch((e) => { | ||
throw `A ${ | ||
e.statusCode | ||
} error occurred when trying to fetch dspa aws storage credentials: ${ | ||
e.response?.body?.message || e?.response?.statusMessage | ||
}`; | ||
}); | ||
|
||
const dspas = ( | ||
dspaResponse?.body as { | ||
items: DSPipelineKind[]; | ||
} | ||
)?.items; | ||
|
||
if (!dspas || !dspas.length) { | ||
throw 'No Data Science Pipeline Application found'; | ||
} | ||
|
||
// check if data connection is available | ||
if ( | ||
dspas[0].status.conditions.find((condition) => condition.type === 'ObjectStoreAvailable') | ||
.status !== 'True' | ||
) { | ||
throw 'Object store is not available'; | ||
} | ||
|
||
// always get the first one | ||
const externalStorage = dspas[0].spec.objectStorage.externalStorage; | ||
|
||
if (externalStorage) { | ||
const { region, host: endPoint, s3CredentialsSecret } = externalStorage; | ||
|
||
// get secret | ||
const secret = await fastify.kube.coreV1Api.readNamespacedSecret( | ||
s3CredentialsSecret.secretName, | ||
namespace, | ||
undefined, | ||
undefined, | ||
undefined, | ||
{ | ||
headers: { | ||
authorization: `Bearer ${token}`, | ||
}, | ||
}, | ||
); | ||
const accessKey = atob(secret.body.data[s3CredentialsSecret.accessKey]); | ||
const secretKey = atob(secret.body.data[s3CredentialsSecret.secretKey]); | ||
|
||
if (!accessKey || !secretKey) { | ||
throw 'Access key or secret key is empty'; | ||
} | ||
|
||
// sessionToken | ||
return new MinioClient({ accessKey, secretKey, endPoint, region }); | ||
} | ||
} catch (err) { | ||
console.error('Unable to create minio client: ', err); | ||
throw new Error('Unable to create minio client: ' + err); | ||
} | ||
} | ||
|
||
/** MinioRequestConfig describes the info required to retrieve an artifact. */ | ||
export interface MinioRequestConfig { | ||
bucket: string; | ||
key: string; | ||
client: MinioClient; | ||
} | ||
|
||
/** | ||
* Returns a stream from an object in a s3 compatible object store (e.g. minio). | ||
* | ||
* @param param.bucket Bucket name to retrieve the object from. | ||
* @param param.key Key of the object to retrieve. | ||
* @param param.client Minio client. | ||
* | ||
*/ | ||
export async function getObjectStream({ | ||
bucket, | ||
key, | ||
client, | ||
}: MinioRequestConfig): Promise<Transform> { | ||
const stream = await client.getObject(bucket, key); | ||
return stream.pipe(new PassThrough()); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.