kubernetesApiClient.ts 5.29 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
/**
 * Copyright (c) Microsoft Corporation
 * All rights reserved.
 *
 * MIT License
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated
 * documentation files (the "Software"), to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and
 * to permit persons to whom the Software is furnished to do so, subject to the following conditions:
 * The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED *AS IS*, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
 * BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */

'use strict';

import * as os from 'os'
import * as path from 'path';
import { getLogger, Logger } from '../../common/log';

var K8SClient = require('kubernetes-client').Client;
var K8SConfig = require('kubernetes-client').config;

/**
 * Generict Kubernetes client, target version >= 1.9
 */
class GeneralK8sClient {
    protected readonly client: any;
    protected readonly log: Logger = getLogger();

    constructor() {
37
        this.client = new K8SClient({ config: K8SConfig.fromKubeconfig(), version: '1.9'});
38
39
40
41
        this.client.loadSpec();
    }

    public async createSecret(secretManifest: any): Promise<boolean> {
42
        let result: Promise<boolean>;
43
44
45
46
47
48
49
50
51
52
        const response : any = await this.client.api.v1.namespaces('default').secrets.post({body: secretManifest});
        if(response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
            result = Promise.resolve(true);
        } else {
            result = Promise.reject(`Create secrets failed, statusCode is ${response.statusCode}`);
        }
        return result;
    }
}

53
abstract class KubernetesCRDClient {
54
55
56
57
58
    protected readonly client: any;
    protected readonly log: Logger = getLogger();
    protected crdSchema: any;

    constructor() {
59
        this.client = new K8SClient({ config: K8SConfig.fromKubeconfig() });
60
61
62
63
64
65
66
67
        this.client.loadSpec();
    }

    protected abstract get operator(): any;

    public abstract get containerName(): string;

    public get jobKind(): string {
68
69
        if(this.crdSchema
            && this.crdSchema.spec
70
71
72
73
74
75
76
77
78
            && this.crdSchema.spec.names
            && this.crdSchema.spec.names.kind) {
            return this.crdSchema.spec.names.kind;
        } else {
            throw new Error('KubeflowOperatorClient: getJobKind failed, kind is undefined in crd schema!');
        }
    }

    public get apiVersion(): string {
79
80
        if(this.crdSchema
            && this.crdSchema.spec
81
82
83
84
85
86
            && this.crdSchema.spec.version) {
            return this.crdSchema.spec.version;
        } else {
            throw new Error('KubeflowOperatorClient: get apiVersion failed, version is undefined in crd schema!');
        }
    }
87

88
    public async createKubernetesJob(jobManifest: any): Promise<boolean> {
89
90
91
92
93
        let result: Promise<boolean>;
        const response : any = await this.operator.post({body: jobManifest});
        if(response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
            result = Promise.resolve(true);
        } else {
94
            result = Promise.reject(`Create kubernetes job failed, statusCode is ${response.statusCode}`);
95
96
97
98
99
        }
        return result;
    }

    //TODO : replace any
100
    public async getKubernetesJob(kubeflowJobName: string): Promise<any> {
101
102
103
104
105
106
107
108
109
110
        let result: Promise<any>;
        const response : any = await this.operator(kubeflowJobName).get();
        if(response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
            result = Promise.resolve(response.body);
        } else {
            result = Promise.reject(`KubeflowOperatorClient get tfjobs failed, statusCode is ${response.statusCode}`);
        }
        return result;
    }

111
    public async deleteKubernetesJob(labels: Map<string, string>): Promise<boolean> {
112
113
114
115
        let result: Promise<boolean>;
        // construct match query from labels for deleting tfjob
        const matchQuery: string = Array.from(labels.keys()).map(labelKey => `${labelKey}=${labels.get(labelKey)}`).join(',');
        try {
116
117
118
119
            const deleteResult : any = await this.operator().delete({
                 qs: {
                      labelSelector: matchQuery,
                      propagationPolicy: "Background"
120
                     }
121
            });
122
123
124
125
126
127
128
129
130
131
132
133
134
            if(deleteResult.statusCode && deleteResult.statusCode >= 200 && deleteResult.statusCode <= 299) {
                result = Promise.resolve(true);
            } else {
                result = Promise.reject(`KubeflowOperatorClient, delete labels ${matchQuery} get wrong statusCode ${deleteResult.statusCode}`);
            }
        } catch(err) {
            result = Promise.reject(err);
        }

        return result;
    }
}

135
export { KubernetesCRDClient, GeneralK8sClient };