util.ts 7.14 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
import { getLogger } from "common/log";

/**
 * 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 { countFilesRecursively } from '../../common/utils'
25
26
import * as cpp from 'child-process-promise';
import * as cp from 'child_process';
27
28
29
30
import * as os from 'os';
import * as fs from 'fs';
import { getNewLine } from '../../common/utils';
import { GPU_INFO_COLLECTOR_FORMAT_LINUX, GPU_INFO_COLLECTOR_FORMAT_WINDOWS } from './gpuData';
31
32
33
import * as path from 'path';
import { String } from 'typescript-string-operations';
import { file } from "../../node_modules/@types/tmp";
34
35
36

/**
 * Validate codeDir, calculate file count recursively under codeDir, and throw error if any rule is broken
37
 *
38
39
40
41
42
43
44
45
46
47
48
49
50
 * @param codeDir codeDir in nni config file
 * @returns file number under codeDir
 */
export async function validateCodeDir(codeDir: string) : Promise<number> {
    let fileCount: number | undefined;

    try {
        fileCount = await countFilesRecursively(codeDir);
    } catch(error) {
        throw new Error(`Call count file error: ${error}`);
    }

    if(fileCount && fileCount > 1000) {
51
        const errMessage: string = `Too many files(${fileCount} found}) in ${codeDir},`
52
                                    + ` please check if it's a valid code dir`;
53
        throw new Error(errMessage);
54
55
56
    }

    return fileCount;
57
58
59
60
}

/**
 * crete a new directory
61
 * @param directory
62
63
64
65
66
67
68
69
70
71
 */
export async function execMkdir(directory: string): Promise<void> {
    if (process.platform === 'win32') {
        await cpp.exec(`powershell.exe New-Item -Path ${directory} -ItemType "directory" -Force`);
    } else {
        await cpp.exec(`mkdir -p ${directory}`);
    }
    return Promise.resolve();
}

72
73
74
75
76
77
78
79
80
81
82
83
84
85
/**
 * copy files to the directory
 * @param source
 * @param destination
 */
export async function execCopydir(source: string, destination: string): Promise<void> {
    if (process.platform === 'win32') {
        await cpp.exec(`powershell.exe Copy-Item ${source} -Destination ${destination} -Recurse`);
    } else {
        await cpp.exec(`cp -r ${source} ${destination}`);
    }
    return Promise.resolve();
}

86
87
/**
 * crete a new file
88
 * @param filename
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
 */
export async function execNewFile(filename: string): Promise<void> {
    if (process.platform === 'win32') {
        await cpp.exec(`powershell.exe New-Item -Path ${filename} -ItemType "file" -Force`);
    } else {
        await cpp.exec(`touch ${filename}`);
    }
    return Promise.resolve();
}

/**
 * run script
 * @param filePath
 */
export function execScript(filePath: string): cp.ChildProcess {
    if (process.platform === 'win32') {
105
        return cp.exec(`powershell.exe -ExecutionPolicy Bypass -file ${filePath}`);
106
107
108
109
110
111
112
    } else {
        return cp.exec(`bash ${filePath}`);
    }
}

/**
 * output the last line of a file
113
 * @param filePath
114
115
116
117
118
119
120
121
122
123
124
125
126
 */
export async function execTail(filePath: string): Promise<cpp.childProcessPromise.Result> {
    let cmdresult: cpp.childProcessPromise.Result;
    if (process.platform === 'win32') {
        cmdresult = await cpp.exec(`powershell.exe Get-Content ${filePath} -Tail 1`);
    } else {
        cmdresult = await cpp.exec(`tail -n 1 ${filePath}`);
    }
    return Promise.resolve(cmdresult);
}

/**
 * delete a directory
127
 * @param directory
128
 */
129
export async function execRemove(directory: string): Promise<void> {
130
    if (process.platform === 'win32') {
131
        await cpp.exec(`powershell.exe Remove-Item ${directory} -Recurse -Force`);
132
133
134
135
136
137
138
139
    } else {
        await cpp.exec(`rm -rf ${directory}`);
    }
    return Promise.resolve();
}

/**
 * kill a process
140
 * @param directory
141
 */
142
export async function execKill(pid: string): Promise<void> {
143
144
145
146
147
148
149
150
151
152
153
    if (process.platform === 'win32') {
        await cpp.exec(`cmd /c taskkill /PID ${pid} /T /F`);
    } else {
        await cpp.exec(`pkill -P ${pid}`);
    }
    return Promise.resolve();
}

/**
 * set environment variable
 * @param  variable
154
 * @returns command string
155
 */
156
export function setEnvironmentVariable(variable: { key: string; value: string }): string {
157
158
159
160
161
162
163
164
    if (process.platform === 'win32') {
        return `$env:${variable.key}="${variable.value}"`;
    }
    else{
        return `export ${variable.key}=${variable.value}`;
    }
}

165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
/**
 * Compress files in directory to tar file
 * @param  source_path
 * @param  tar_path
 */
export async function tarAdd(tar_path: string, source_path: string): Promise<void> {
    if (process.platform === 'win32') {
        tar_path = tar_path.split('\\').join('\\\\');
        source_path = source_path.split('\\').join('\\\\');
        let script: string[] = [];
        script.push(
            `import os`,
            `import tarfile`,
            String.Format(`tar = tarfile.open("{0}","w:gz")\r\nfor root,dir,files in os.walk("{1}"):`, tar_path, source_path),
            `    for file in files:`,
            `        fullpath = os.path.join(root,file)`,
            `        tar.add(fullpath, arcname=file)`,
            `tar.close()`);
        await fs.promises.writeFile(path.join(os.tmpdir(), 'tar.py'), script.join(getNewLine()), { encoding: 'utf8', mode: 0o777 });
        const tarScript: string = path.join(os.tmpdir(), 'tar.py');
        await cpp.exec(`python ${tarScript}`);
    } else {
        await cpp.exec(`tar -czf ${tar_path} -C ${source_path} .`);
    }
    return Promise.resolve();
}
191
192
193

/**
 * generate script file name
194
 * @param fileNamePrefix
195
196
197
198
199
200
201
202
203
204
205
 */
export function getScriptName(fileNamePrefix: string): string {
    if (process.platform === 'win32') {
        return fileNamePrefix + '.ps1';
    } else {
        return fileNamePrefix + '.sh';
    }
}

/**
 * generate script file
206
 * @param gpuMetricCollectorScriptFolder
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
 */
export function getgpuMetricsCollectorScriptContent(gpuMetricCollectorScriptFolder: string): string {
    if(process.platform === 'win32') {
        return String.Format(
            GPU_INFO_COLLECTOR_FORMAT_WINDOWS,
            gpuMetricCollectorScriptFolder,
            path.join(gpuMetricCollectorScriptFolder, 'pid'),
        );
    } else {
        return String.Format(
            GPU_INFO_COLLECTOR_FORMAT_LINUX,
            gpuMetricCollectorScriptFolder,
            path.join(gpuMetricCollectorScriptFolder, 'pid'),
        );
    }
}