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

'use strict';

6
7
import * as azureStorage from 'azure-storage';
import * as fs from 'fs';
SparkSnail's avatar
SparkSnail committed
8
9
import * as path from 'path';
import { Deferred } from 'ts-deferred';
10
import { String } from 'typescript-string-operations';
SparkSnail's avatar
SparkSnail committed
11
12
13
import { getLogger } from '../../common/log';
import { mkDirP } from '../../common/utils';

14
// tslint:disable: no-redundant-jsdoc no-any no-unsafe-any
SparkSnail's avatar
SparkSnail committed
15
export namespace AzureStorageClientUtility {
16

SparkSnail's avatar
SparkSnail committed
17
18
    /**
     * create azure share
19
20
     * @param fileServerClient
     * @param azureShare
SparkSnail's avatar
SparkSnail committed
21
     */
22
23
    export async function createShare(fileServerClient: any, azureShare: any): Promise<boolean> {
        const deferred: Deferred<boolean> = new Deferred<boolean>();
24
25
26
27
        fileServerClient.createShareIfNotExists(azureShare, (error: any, result: any, response: any) => {
            if (error) {
                getLogger()
                  .error(`Create share failed:, ${error}`);
28
                deferred.resolve(false);
29
            } else {
30
                deferred.resolve(true);
SparkSnail's avatar
SparkSnail committed
31
            }
32
33
        });

SparkSnail's avatar
SparkSnail committed
34
35
        return deferred.promise;
    }
36

SparkSnail's avatar
SparkSnail committed
37
38
    /**
     * Create a new directory (NOT recursively) in azure file storage.
39
40
41
     * @param fileServerClient
     * @param azureFoler
     * @param azureShare
SparkSnail's avatar
SparkSnail committed
42
     */
43
44
    export async function createDirectory(fileServerClient: azureStorage.FileService, azureFoler: any, azureShare: any): Promise<boolean> {
        const deferred: Deferred<boolean> = new Deferred<boolean>();
45
46
47
48
        fileServerClient.createDirectoryIfNotExists(azureShare, azureFoler, (error: any, result: any, response: any) => {
            if (error) {
                getLogger()
                  .error(`Create directory failed:, ${error}`);
49
                deferred.resolve(false);
50
            } else {
51
                deferred.resolve(true);
SparkSnail's avatar
SparkSnail committed
52
            }
53
        });
SparkSnail's avatar
SparkSnail committed
54
55
56
57
58
59
        return deferred.promise;
    }

    /**
     * Create a new directory recursively in azure file storage
     * @param fileServerClient
60
     * @param azureDirectory
SparkSnail's avatar
SparkSnail committed
61
     */
62
    export async function createDirectoryRecursive(fileServerClient: azureStorage.FileService, azureDirectory: string,
63
64
                                                   azureShare: any): Promise<boolean> {
        const deferred: Deferred<boolean> = new Deferred<boolean>();
65
66
67
        const directories: string[] = azureDirectory.split('/');
        let rootDirectory: string = '';
        for (const directory of directories) {
SparkSnail's avatar
SparkSnail committed
68
            rootDirectory += directory;
69
70
71
72
73
            let result:boolean = await createDirectory(fileServerClient, rootDirectory, azureShare);
            if (!result) {
                deferred.resolve(false);
                return deferred.promise;
            }
SparkSnail's avatar
SparkSnail committed
74
75
            rootDirectory += '/';
        }
76
        deferred.resolve(true);
77

SparkSnail's avatar
SparkSnail committed
78
79
        return deferred.promise;
    }
80

SparkSnail's avatar
SparkSnail committed
81
82
    /**
     * upload a file to azure storage
83
84
85
86
87
     * @param fileServerClient
     * @param azureDirectory
     * @param azureFileName
     * @param azureShare
     * @param localFilePath
SparkSnail's avatar
SparkSnail committed
88
     */
89
    async function uploadFileToAzure(fileServerClient: any, azureDirectory: string, azureFileName: any, azureShare: any,
90
91
                                     localFilePath: string): Promise<boolean> {
        const deferred: Deferred<boolean> = new Deferred<boolean>();
92
93
94
95
96
        await fileServerClient.createFileFromLocalFile(azureShare, azureDirectory, azureFileName, localFilePath,
                                                       (error: any, result: any, response: any) => {
            if (error) {
                getLogger()
                  .error(`Upload file failed:, ${error}`);
97
                deferred.resolve(false);
98
            } else {
99
                deferred.resolve(true);
SparkSnail's avatar
SparkSnail committed
100
            }
101
102
        });

SparkSnail's avatar
SparkSnail committed
103
104
        return deferred.promise;
    }
105

SparkSnail's avatar
SparkSnail committed
106
107
    /**
     * download a file from azure storage
108
109
110
111
112
     * @param fileServerClient
     * @param azureDirectory
     * @param azureFileName
     * @param azureShare
     * @param localFilePath
SparkSnail's avatar
SparkSnail committed
113
     */
114
    async function downloadFile(fileServerClient: any, azureDirectory: string, azureFileName: any, azureShare: any,
115
116
                                localFilePath: string): Promise<boolean> {
        const deferred: Deferred<boolean> = new Deferred<boolean>();
117
118
119
120
121
122
        // tslint:disable-next-line:non-literal-fs-path
        await fileServerClient.getFileToStream(azureShare, azureDirectory, azureFileName, fs.createWriteStream(localFilePath),
                                               (error: any, result: any, response: any) => {
            if (error) {
                getLogger()
                  .error(`Download file failed:, ${error}`);
123
                deferred.resolve(false);
124
            } else {
125
                deferred.resolve(true);
SparkSnail's avatar
SparkSnail committed
126
            }
127
128
        });

SparkSnail's avatar
SparkSnail committed
129
130
131
132
133
134
135
136
137
138
        return deferred.promise;
    }

    /**
     * Upload a directory to azure file storage
     * @param fileServerClient : the client of file server
     * @param azureDirectory : the directory in azure file storage
     * @param azureShare : the azure share used
     * @param localDirectory : local directory to be uploaded
     */
139
140
    // tslint:disable:non-literal-fs-path
    export async function uploadDirectory(fileServerClient: azureStorage.FileService, azureDirectory: string, azureShare: any,
141
142
                                          localDirectory: string): Promise<boolean> {
        const deferred: Deferred<boolean> = new Deferred<boolean>();
SparkSnail's avatar
SparkSnail committed
143
        const fileNameArray: string[] = fs.readdirSync(localDirectory);
144
145
146
147
148
        let result: boolean = await createDirectoryRecursive(fileServerClient, azureDirectory, azureShare);
        if (!result) {
            deferred.resolve(false);
            return deferred.promise;
        }
149
        for (const fileName of fileNameArray) {
SparkSnail's avatar
SparkSnail committed
150
151
            const fullFilePath: string = path.join(localDirectory, fileName);
            try {
152
153
                let resultUploadFile: boolean = true;
                let resultUploadDir: boolean = true;
154
155
                if (fs.lstatSync(fullFilePath)
                      .isFile()) {
156
                    resultUploadFile = await uploadFileToAzure(fileServerClient, azureDirectory, fileName, azureShare, fullFilePath);
SparkSnail's avatar
SparkSnail committed
157
158
                } else {
                    // If filePath is a directory, recuisively copy it to azure
159
160
161
162
163
                    resultUploadDir = await uploadDirectory(fileServerClient, String.Format('{0}/{1}', azureDirectory, fileName), azureShare, fullFilePath);
                }
                if (!(resultUploadFile && resultUploadDir)) {
                    deferred.resolve(false);
                    return deferred.promise;
SparkSnail's avatar
SparkSnail committed
164
                }
165
            } catch (error) {
166
                deferred.resolve(false);
167

SparkSnail's avatar
SparkSnail committed
168
169
170
171
                return deferred.promise;
            }
        }
        // All files/directories are copied successfully, resolve
172
        deferred.resolve(true);
173

SparkSnail's avatar
SparkSnail committed
174
175
        return deferred.promise;
    }
176

SparkSnail's avatar
SparkSnail committed
177
178
    /**
     * downlod a directory from azure
179
180
181
182
     * @param fileServerClient
     * @param azureDirectory
     * @param azureShare
     * @param localDirectory
SparkSnail's avatar
SparkSnail committed
183
     */
184
185
    export async function downloadDirectory(fileServerClient: any, azureDirectory: string, azureShare: any, localDirectory: string):
     Promise<void> {
SparkSnail's avatar
SparkSnail committed
186
        const deferred: Deferred<void> = new Deferred<void>();
187
188
189
190
191
192
        await mkDirP(localDirectory);
        fileServerClient.listFilesAndDirectoriesSegmented(azureShare, azureDirectory, 'null',
                                                          async (error: any, result: any, response: any) => {
            if (('entries' in result) === false) {
                getLogger()
                  .error(`list files failed, can't get entries in result`);
SparkSnail's avatar
SparkSnail committed
193
194
195
                throw new Error(`list files failed, can't get entries in result`);
            }

196
197
198
            if (('files' in result.entries) === false) {
                getLogger()
                  .error(`list files failed, can't get files in result['entries']`);
SparkSnail's avatar
SparkSnail committed
199
200
201
                throw new Error(`list files failed, can't get files in result['entries']`);
            }

202
203
204
            if (('directories' in result.directories) === false) {
                getLogger()
                  .error(`list files failed, can't get directories in result['entries']`);
SparkSnail's avatar
SparkSnail committed
205
206
207
                throw new Error(`list files failed, can't get directories in result['entries']`);
            }

208
            for (const fileName of result.entries.files) {
SparkSnail's avatar
SparkSnail committed
209
                const fullFilePath: string = path.join(localDirectory, fileName.name);
210
                await downloadFile(fileServerClient, azureDirectory, fileName.name, azureShare, fullFilePath);
SparkSnail's avatar
SparkSnail committed
211
            }
212

213
214
215
216
            for (const directoryName of result.entries.directories) {
                const fullDirectoryPath: string = path.join(localDirectory, directoryName.name);
                const fullAzureDirectory: string = path.join(azureDirectory, directoryName.name);
                await downloadDirectory(fileServerClient, fullAzureDirectory, azureShare, fullDirectoryPath);
SparkSnail's avatar
SparkSnail committed
217
218
            }
            deferred.resolve();
219
220
        });

SparkSnail's avatar
SparkSnail committed
221
222
223
        return deferred.promise;
    }
}
224
// tslint:enable: no-redundant-jsdoc no-any no-unsafe-any