kubernetesApiClient.ts 8.16 KB
Newer Older
liuzhe-lz's avatar
liuzhe-lz committed
1
2
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT license.
3
4
5

'use strict';

chicm-ms's avatar
chicm-ms committed
6
// eslint-disable-next-line @typescript-eslint/camelcase
7
import { Client1_10, config } from 'kubernetes-client';
8
9
10
11
12
13
14
15
16
17
import { getLogger, Logger } from '../../common/log';

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

    constructor() {
18
        this.client = new Client1_10({ config: config.fromKubeconfig(), version: '1.9'});
19
20
21
        this.client.loadSpec();
    }

22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
    private matchStorageClass(response: any): string {
        const adlSupportedProvisioners: RegExp[] = [
            new RegExp("microk8s.io/hostpath"),
            new RegExp(".*cephfs.csi.ceph.com"),
            new RegExp(".*azure.*"),
            new RegExp("\\b" + "efs" + "\\b")
        ]
        const templateLen = adlSupportedProvisioners.length,
              responseLen = response.items.length
        let i = 0,
            j = 0;
        for (; i < responseLen; i++) {
            const provisioner: string = response.items[i].provisioner
            for (; j < templateLen; j++) {
                if (provisioner.match(adlSupportedProvisioners[j])) {
                    return response.items[i].metadata.name;
                }
            }
        }
        return "Not Found!";
    }

    public async getStorageClass(): Promise<string> {
        let result: Promise<string>;
        const response: any = await this.client.apis["storage.k8s.io"].v1beta1.storageclasses.get()
        const storageClassType: string = this.matchStorageClass(response.body)
        if (response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
            if (storageClassType != "Not Found!") {
                result = Promise.resolve(storageClassType);
            }
            else {
                result = Promise.reject("No StorageClasses are supported!")
            }
        } else {
            result = Promise.reject(`List storageclasses failed, statusCode is ${response.statusCode}`);
        }
        return result;
    }

    public async createDeployment(deploymentManifest: any): Promise<string> {
        let result: Promise<string>;
        const response: any = await this.client.apis.apps.v1.namespaces('default').deployments.post({ body: deploymentManifest })
        if (response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
            result = Promise.resolve(response.body.metadata.uid);
        } else {
            result = Promise.reject(`Create deployment failed, statusCode is ${response.statusCode}`);
        }
        return result;
    }

    public async deleteDeployment(deploymentName: string): Promise<boolean> {
        let result: Promise<boolean>;
        // TODO: change this hard coded deployment name after demo
        const response: any = await this.client.apis.apps.v1.namespaces('default')
          .deployment(deploymentName).delete();
        if (response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
            result = Promise.resolve(true);
        } else {
            result = Promise.reject(`Delete deployment failed, statusCode is ${response.statusCode}`);
        }
        return result;
    }

    public async createConfigMap(configMapManifest: any): Promise<boolean> {
        let result: Promise<boolean>;
        const response: any = await this.client.api.v1.namespaces('default')
          .configmaps.post({body: configMapManifest});
        if (response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
            result = Promise.resolve(true);
        } else {
            result = Promise.reject(`Create configMap failed, statusCode is ${response.statusCode}`);
        }

        return result;
    }

    public async createPersistentVolumeClaim(pvcManifest: any): Promise<boolean> {
        let result: Promise<boolean>;
        const response: any = await this.client.api.v1.namespaces('default')
          .persistentvolumeclaims.post({body: pvcManifest});
        if (response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
            result = Promise.resolve(true);
        } else {
            result = Promise.reject(`Create pvc failed, statusCode is ${response.statusCode}`);
        }
        return result;
    }

110
    public async createSecret(secretManifest: any): Promise<boolean> {
111
        let result: Promise<boolean>;
chicm-ms's avatar
chicm-ms committed
112
        const response: any = await this.client.api.v1.namespaces('default').secrets
113
114
          .post({body: secretManifest});
        if (response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
115
116
117
118
            result = Promise.resolve(true);
        } else {
            result = Promise.reject(`Create secrets failed, statusCode is ${response.statusCode}`);
        }
119

120
121
122
123
        return result;
    }
}

124
125
126
/**
 * Kubernetes CRD client
 */
127
abstract class KubernetesCRDClient {
128
129
130
131
132
    protected readonly client: any;
    protected readonly log: Logger = getLogger();
    protected crdSchema: any;

    constructor() {
133
        this.client = new Client1_10({ config: config.fromKubeconfig() });
134
135
136
137
138
139
140
141
        this.client.loadSpec();
    }

    protected abstract get operator(): any;

    public abstract get containerName(): string;

    public get jobKind(): string {
142
        if (this.crdSchema
143
            && this.crdSchema.spec
144
145
146
147
148
149
150
151
152
            && 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 {
153
        if (this.crdSchema
154
            && this.crdSchema.spec
155
156
157
158
159
160
            && this.crdSchema.spec.version) {
            return this.crdSchema.spec.version;
        } else {
            throw new Error('KubeflowOperatorClient: get apiVersion failed, version is undefined in crd schema!');
        }
    }
161

162
    public async createKubernetesJob(jobManifest: any): Promise<boolean> {
163
        let result: Promise<boolean>;
chicm-ms's avatar
chicm-ms committed
164
        const response: any = await this.operator.post({body: jobManifest});
165
        if (response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
166
167
            result = Promise.resolve(true);
        } else {
168
            result = Promise.reject(`KubernetesApiClient createKubernetesJob failed, statusCode is ${response.statusCode}`);
169
        }
170

171
172
173
174
        return result;
    }

    //TODO : replace any
175
    public async getKubernetesJob(kubeflowJobName: string): Promise<any> {
176
        let result: Promise<any>;
chicm-ms's avatar
chicm-ms committed
177
        const response: any = await this.operator(kubeflowJobName)
178
179
          .get();
        if (response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
180
181
            result = Promise.resolve(response.body);
        } else {
182
            result = Promise.reject(`KubernetesApiClient getKubernetesJob failed, statusCode is ${response.statusCode}`);
183
        }
184

185
186
187
        return result;
    }

188
    public async deleteKubernetesJob(labels: Map<string, string>): Promise<boolean> {
189
190
        let result: Promise<boolean>;
        // construct match query from labels for deleting tfjob
191
192
193
        const matchQuery: string = Array.from(labels.keys())
                                     .map((labelKey: string) => `${labelKey}=${labels.get(labelKey)}`)
                                     .join(',');
194
        try {
chicm-ms's avatar
chicm-ms committed
195
            const deleteResult: any = await this.operator()
196
              .delete({
197
198
                 qs: {
                      labelSelector: matchQuery,
199
                      propagationPolicy: 'Background'
200
                     }
201
            });
202
            if (deleteResult.statusCode && deleteResult.statusCode >= 200 && deleteResult.statusCode <= 299) {
203
204
                result = Promise.resolve(true);
            } else {
205
                result = Promise.reject(
206
                    `KubernetesApiClient, delete labels ${matchQuery} get wrong statusCode ${deleteResult.statusCode}`);
207
            }
208
        } catch (err) {
209
210
211
212
213
214
215
            result = Promise.reject(err);
        }

        return result;
    }
}

216
export { KubernetesCRDClient, GeneralK8sClient };