kubernetesApiClient.ts 8.33 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
import { getLogger, Logger } from '../../common/log';

/**
11
 * Generic Kubernetes client, target version >= 1.9
12
13
14
15
 */
class GeneralK8sClient {
    protected readonly client: any;
    protected readonly log: Logger = getLogger();
16
    protected namespace: string = 'default';
17
18

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

23
24
25
26
    public set setNamespace(namespace: string) {
        this.namespace = namespace;
    }

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
    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>;
68
69
        const response: any = await this.client.apis.apps.v1.namespaces(this.namespace)
          .deployments.post({ body: deploymentManifest })
70
71
72
73
74
75
76
77
78
79
80
        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
81
        const response: any = await this.client.apis.apps.v1.namespaces(this.namespace)
82
83
84
85
86
87
88
89
90
91
92
          .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>;
93
        const response: any = await this.client.api.v1.namespaces(this.namespace)
94
95
96
97
98
99
100
101
102
103
104
105
          .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>;
106
        const response: any = await this.client.api.v1.namespaces(this.namespace)
107
108
109
110
111
112
113
114
115
          .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;
    }

116
    public async createSecret(secretManifest: any): Promise<boolean> {
117
        let result: Promise<boolean>;
118
119
        const response: any = await this.client.api.v1.namespaces(this.namespace)
          .secrets.post({body: secretManifest});
120
        if (response.statusCode && (response.statusCode >= 200 && response.statusCode <= 299)) {
121
122
123
124
            result = Promise.resolve(true);
        } else {
            result = Promise.reject(`Create secrets failed, statusCode is ${response.statusCode}`);
        }
125

126
127
128
129
        return result;
    }
}

130
131
132
/**
 * Kubernetes CRD client
 */
133
abstract class KubernetesCRDClient {
134
135
136
137
138
    protected readonly client: any;
    protected readonly log: Logger = getLogger();
    protected crdSchema: any;

    constructor() {
139
        this.client = new Client1_10({ config: config.fromKubeconfig() });
140
141
142
143
144
145
146
147
        this.client.loadSpec();
    }

    protected abstract get operator(): any;

    public abstract get containerName(): string;

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

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

177
178
179
180
        return result;
    }

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

191
192
193
        return result;
    }

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

        return result;
    }
}

222
export { KubernetesCRDClient, GeneralK8sClient };