UNPKG

44.7 kBJavaScriptView Raw
1"use strict";
2var _a, _b;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.DockerVolumeConsistency = exports.DockerImage = exports.BundlingDockerImage = exports.BundlingOutput = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const child_process_1 = require("child_process");
8const crypto = require("crypto");
9const path_1 = require("path");
10const fs_1 = require("./fs");
11const jsii_deprecated_1 = require("./private/jsii-deprecated");
12/**
13 * The type of output that a bundling operation is producing.
14 *
15 */
16var BundlingOutput;
17(function (BundlingOutput) {
18 /**
19 * The bundling output directory includes a single .zip or .jar file which
20 * will be used as the final bundle. If the output directory does not
21 * include exactly a single archive, bundling will fail.
22 */
23 BundlingOutput["ARCHIVED"] = "archived";
24 /**
25 * The bundling output directory contains one or more files which will be
26 * archived and uploaded as a .zip file to S3.
27 */
28 BundlingOutput["NOT_ARCHIVED"] = "not-archived";
29 /**
30 * If the bundling output directory contains a single archive file (zip or jar)
31 * it will be used as the bundle output as-is. Otherwise all the files in the bundling output directory will be zipped.
32 */
33 BundlingOutput["AUTO_DISCOVER"] = "auto-discover";
34})(BundlingOutput = exports.BundlingOutput || (exports.BundlingOutput = {}));
35/**
36 * A Docker image used for asset bundling
37 *
38 * @deprecated use DockerImage
39 */
40class BundlingDockerImage {
41 /** @param image The Docker image */
42 constructor(image, _imageHash) {
43 this.image = image;
44 this._imageHash = _imageHash;
45 try {
46 jsiiDeprecationWarnings.print("@aws-cdk/core.BundlingDockerImage", "use DockerImage");
47 }
48 catch (error) {
49 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
50 Error.captureStackTrace(error, BundlingDockerImage);
51 }
52 throw error;
53 }
54 }
55 /**
56 * Reference an image on DockerHub or another online registry.
57 *
58 * @param image the image name
59 */
60 static fromRegistry(image) {
61 try {
62 jsiiDeprecationWarnings.print("@aws-cdk/core.BundlingDockerImage#fromRegistry", "use DockerImage");
63 }
64 catch (error) {
65 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
66 Error.captureStackTrace(error, this.fromRegistry);
67 }
68 throw error;
69 }
70 return new DockerImage(image);
71 }
72 /**
73 * Reference an image that's built directly from sources on disk.
74 *
75 * @param path The path to the directory containing the Docker file
76 * @param options Docker build options
77 *
78 * @deprecated use DockerImage.fromBuild()
79 */
80 static fromAsset(path, options = {}) {
81 try {
82 jsiiDeprecationWarnings.print("@aws-cdk/core.BundlingDockerImage#fromAsset", "use DockerImage.fromBuild()");
83 jsiiDeprecationWarnings._aws_cdk_core_DockerBuildOptions(options);
84 }
85 catch (error) {
86 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
87 Error.captureStackTrace(error, this.fromAsset);
88 }
89 throw error;
90 }
91 return DockerImage.fromBuild(path, options);
92 }
93 /**
94 * Provides a stable representation of this image for JSON serialization.
95 *
96 * @return The overridden image name if set or image hash name in that order
97 */
98 toJSON() {
99 try {
100 jsiiDeprecationWarnings.print("@aws-cdk/core.BundlingDockerImage#toJSON", "use DockerImage");
101 }
102 catch (error) {
103 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
104 Error.captureStackTrace(error, this.toJSON);
105 }
106 throw error;
107 }
108 return this._imageHash ?? this.image;
109 }
110 /**
111 * Runs a Docker image
112 */
113 run(options = {}) {
114 try {
115 jsiiDeprecationWarnings.print("@aws-cdk/core.BundlingDockerImage#run", "use DockerImage");
116 jsiiDeprecationWarnings._aws_cdk_core_DockerRunOptions(options);
117 }
118 catch (error) {
119 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
120 Error.captureStackTrace(error, this.run);
121 }
122 throw error;
123 }
124 const volumes = options.volumes || [];
125 const environment = options.environment || {};
126 const entrypoint = options.entrypoint?.[0] || null;
127 const command = [
128 ...options.entrypoint?.[1]
129 ? [...options.entrypoint.slice(1)]
130 : [],
131 ...options.command
132 ? [...options.command]
133 : [],
134 ];
135 const dockerArgs = [
136 'run', '--rm',
137 ...options.securityOpt
138 ? ['--security-opt', options.securityOpt]
139 : [],
140 ...options.user
141 ? ['-u', options.user]
142 : [],
143 ...flatten(volumes.map(v => ['-v', `${v.hostPath}:${v.containerPath}:${isSeLinux() ? 'z,' : ''}${v.consistency ?? DockerVolumeConsistency.DELEGATED}`])),
144 ...flatten(Object.entries(environment).map(([k, v]) => ['--env', `${k}=${v}`])),
145 ...options.workingDirectory
146 ? ['-w', options.workingDirectory]
147 : [],
148 ...entrypoint
149 ? ['--entrypoint', entrypoint]
150 : [],
151 this.image,
152 ...command,
153 ];
154 dockerExec(dockerArgs);
155 }
156 /**
157 * Copies a file or directory out of the Docker image to the local filesystem.
158 *
159 * If `outputPath` is omitted the destination path is a temporary directory.
160 *
161 * @param imagePath the path in the Docker image
162 * @param outputPath the destination path for the copy operation
163 * @returns the destination path
164 */
165 cp(imagePath, outputPath) {
166 try {
167 jsiiDeprecationWarnings.print("@aws-cdk/core.BundlingDockerImage#cp", "use DockerImage");
168 }
169 catch (error) {
170 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
171 Error.captureStackTrace(error, this.cp);
172 }
173 throw error;
174 }
175 const { stdout } = dockerExec(['create', this.image], {}); // Empty options to avoid stdout redirect here
176 const match = stdout.toString().match(/([0-9a-f]{16,})/);
177 if (!match) {
178 throw new Error('Failed to extract container ID from Docker create output');
179 }
180 const containerId = match[1];
181 const containerPath = `${containerId}:${imagePath}`;
182 const destPath = outputPath ?? fs_1.FileSystem.mkdtemp('cdk-docker-cp-');
183 try {
184 dockerExec(['cp', containerPath, destPath]);
185 return destPath;
186 }
187 catch (err) {
188 throw new Error(`Failed to copy files from ${containerPath} to ${destPath}: ${err}`);
189 }
190 finally {
191 dockerExec(['rm', '-v', containerId]);
192 }
193 }
194}
195exports.BundlingDockerImage = BundlingDockerImage;
196_a = JSII_RTTI_SYMBOL_1;
197BundlingDockerImage[_a] = { fqn: "@aws-cdk/core.BundlingDockerImage", version: "1.204.0" };
198/**
199 * A Docker image
200 */
201class DockerImage extends BundlingDockerImage {
202 constructor(image, _imageHash) {
203 // It is preferrable for the deprecated class to inherit a non-deprecated class.
204 // However, in this case, the opposite has occurred which is incompatible with
205 // a deprecation feature. See https://github.com/aws/jsii/issues/3102.
206 const deprecated = jsii_deprecated_1.quiet();
207 super(image, _imageHash);
208 jsii_deprecated_1.reset(deprecated);
209 this.image = image;
210 }
211 /**
212 * Builds a Docker image
213 *
214 * @param path The path to the directory containing the Docker file
215 * @param options Docker build options
216 */
217 static fromBuild(path, options = {}) {
218 try {
219 jsiiDeprecationWarnings._aws_cdk_core_DockerBuildOptions(options);
220 }
221 catch (error) {
222 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
223 Error.captureStackTrace(error, this.fromBuild);
224 }
225 throw error;
226 }
227 const buildArgs = options.buildArgs || {};
228 if (options.file && path_1.isAbsolute(options.file)) {
229 throw new Error(`"file" must be relative to the docker build directory. Got ${options.file}`);
230 }
231 // Image tag derived from path and build options
232 const input = JSON.stringify({ path, ...options });
233 const tagHash = crypto.createHash('sha256').update(input).digest('hex');
234 const tag = `cdk-${tagHash}`;
235 const dockerArgs = [
236 'build', '-t', tag,
237 ...(options.file ? ['-f', path_1.join(path, options.file)] : []),
238 ...(options.platform ? ['--platform', options.platform] : []),
239 ...flatten(Object.entries(buildArgs).map(([k, v]) => ['--build-arg', `${k}=${v}`])),
240 path,
241 ];
242 dockerExec(dockerArgs);
243 // Fingerprints the directory containing the Dockerfile we're building and
244 // differentiates the fingerprint based on build arguments. We do this so
245 // we can provide a stable image hash. Otherwise, the image ID will be
246 // different every time the Docker layer cache is cleared, due primarily to
247 // timestamps.
248 const hash = fs_1.FileSystem.fingerprint(path, { extraHash: JSON.stringify(options) });
249 return new DockerImage(tag, hash);
250 }
251 /**
252 * Reference an image on DockerHub or another online registry.
253 *
254 * @param image the image name
255 */
256 static fromRegistry(image) {
257 return new DockerImage(image);
258 }
259 /**
260 * Provides a stable representation of this image for JSON serialization.
261 *
262 * @return The overridden image name if set or image hash name in that order
263 */
264 toJSON() {
265 // It is preferrable for the deprecated class to inherit a non-deprecated class.
266 // However, in this case, the opposite has occurred which is incompatible with
267 // a deprecation feature. See https://github.com/aws/jsii/issues/3102.
268 const deprecated = jsii_deprecated_1.quiet();
269 const json = super.toJSON();
270 jsii_deprecated_1.reset(deprecated);
271 return json;
272 }
273 /**
274 * Runs a Docker image
275 */
276 run(options = {}) {
277 try {
278 jsiiDeprecationWarnings._aws_cdk_core_DockerRunOptions(options);
279 }
280 catch (error) {
281 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
282 Error.captureStackTrace(error, this.run);
283 }
284 throw error;
285 }
286 // It is preferrable for the deprecated class to inherit a non-deprecated class.
287 // However, in this case, the opposite has occurred which is incompatible with
288 // a deprecation feature. See https://github.com/aws/jsii/issues/3102.
289 const deprecated = jsii_deprecated_1.quiet();
290 const result = super.run(options);
291 jsii_deprecated_1.reset(deprecated);
292 return result;
293 }
294 /**
295 * Copies a file or directory out of the Docker image to the local filesystem.
296 *
297 * If `outputPath` is omitted the destination path is a temporary directory.
298 *
299 * @param imagePath the path in the Docker image
300 * @param outputPath the destination path for the copy operation
301 * @returns the destination path
302 */
303 cp(imagePath, outputPath) {
304 // It is preferrable for the deprecated class to inherit a non-deprecated class.
305 // However, in this case, the opposite has occurred which is incompatible with
306 // a deprecation feature. See https://github.com/aws/jsii/issues/3102.
307 const deprecated = jsii_deprecated_1.quiet();
308 const result = super.cp(imagePath, outputPath);
309 jsii_deprecated_1.reset(deprecated);
310 return result;
311 }
312}
313exports.DockerImage = DockerImage;
314_b = JSII_RTTI_SYMBOL_1;
315DockerImage[_b] = { fqn: "@aws-cdk/core.DockerImage", version: "1.204.0" };
316/**
317 * Supported Docker volume consistency types. Only valid on macOS due to the way file storage works on Mac
318 */
319var DockerVolumeConsistency;
320(function (DockerVolumeConsistency) {
321 /**
322 * Read/write operations inside the Docker container are applied immediately on the mounted host machine volumes
323 */
324 DockerVolumeConsistency["CONSISTENT"] = "consistent";
325 /**
326 * Read/write operations on mounted Docker volumes are first written inside the container and then synchronized to the host machine
327 */
328 DockerVolumeConsistency["DELEGATED"] = "delegated";
329 /**
330 * Read/write operations on mounted Docker volumes are first applied on the host machine and then synchronized to the container
331 */
332 DockerVolumeConsistency["CACHED"] = "cached";
333})(DockerVolumeConsistency = exports.DockerVolumeConsistency || (exports.DockerVolumeConsistency = {}));
334function flatten(x) {
335 return Array.prototype.concat([], ...x);
336}
337function dockerExec(args, options) {
338 const prog = process.env.CDK_DOCKER ?? 'docker';
339 const proc = child_process_1.spawnSync(prog, args, options ?? {
340 stdio: [
341 'ignore',
342 process.stderr,
343 'inherit',
344 ],
345 });
346 if (proc.error) {
347 throw proc.error;
348 }
349 if (proc.status !== 0) {
350 if (proc.stdout || proc.stderr) {
351 throw new Error(`[Status ${proc.status}] stdout: ${proc.stdout?.toString().trim()}\n\n\nstderr: ${proc.stderr?.toString().trim()}`);
352 }
353 throw new Error(`${prog} exited with status ${proc.status}`);
354 }
355 return proc;
356}
357function isSeLinux() {
358 if (process.platform != 'linux') {
359 return false;
360 }
361 const prog = 'selinuxenabled';
362 const proc = child_process_1.spawnSync(prog, [], {
363 stdio: [
364 'pipe',
365 process.stderr,
366 'inherit',
367 ],
368 });
369 if (proc.error) {
370 // selinuxenabled not a valid command, therefore not enabled
371 return false;
372 }
373 if (proc.status == 0) {
374 // selinux enabled
375 return true;
376 }
377 else {
378 // selinux not enabled
379 return false;
380 }
381}
382//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bundling.js","sourceRoot":"","sources":["bundling.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawnSync, SpawnSyncOptions } from 'child_process';\nimport * as crypto from 'crypto';\nimport { isAbsolute, join } from 'path';\nimport { FileSystem } from './fs';\nimport { quiet, reset } from './private/jsii-deprecated';\n\n/**\n * Bundling options\n *\n */\nexport interface BundlingOptions {\n  /**\n   * The Docker image where the command will run.\n   */\n  readonly image: DockerImage;\n\n  /**\n   * The entrypoint to run in the Docker container.\n   *\n   * Example value: `['/bin/sh', '-c']`\n   *\n   * @see https://docs.docker.com/engine/reference/builder/#entrypoint\n   *\n   * @default - run the entrypoint defined in the image\n   */\n  readonly entrypoint?: string[];\n\n  /**\n   * The command to run in the Docker container.\n   *\n   * Example value: `['npm', 'install']`\n   *\n   * @see https://docs.docker.com/engine/reference/run/\n   *\n   * @default - run the command defined in the image\n   */\n  readonly command?: string[];\n\n  /**\n   * Additional Docker volumes to mount.\n   *\n   * @default - no additional volumes are mounted\n   */\n  readonly volumes?: DockerVolume[];\n\n  /**\n   * The environment variables to pass to the Docker container.\n   *\n   * @default - no environment variables.\n   */\n  readonly environment?: { [key: string]: string; };\n\n  /**\n   * Working directory inside the Docker container.\n   *\n   * @default /asset-input\n   */\n  readonly workingDirectory?: string;\n\n  /**\n   * The user to use when running the Docker container.\n   *\n   *   user | user:group | uid | uid:gid | user:gid | uid:group\n   *\n   * @see https://docs.docker.com/engine/reference/run/#user\n   *\n   * @default - uid:gid of the current user or 1000:1000 on Windows\n   */\n  readonly user?: string;\n\n  /**\n   * Local bundling provider.\n   *\n   * The provider implements a method `tryBundle()` which should return `true`\n   * if local bundling was performed. If `false` is returned, docker bundling\n   * will be done.\n   *\n   * @default - bundling will only be performed in a Docker container\n   *\n   */\n  readonly local?: ILocalBundling;\n\n  /**\n   * The type of output that this bundling operation is producing.\n   *\n   * @default BundlingOutput.AUTO_DISCOVER\n   *\n   */\n  readonly outputType?: BundlingOutput;\n\n  /**\n   * [Security configuration](https://docs.docker.com/engine/reference/run/#security-configuration)\n   * when running the docker container.\n   *\n   * @default - no security options\n   */\n  readonly securityOpt?: string;\n}\n\n/**\n * The type of output that a bundling operation is producing.\n *\n */\nexport enum BundlingOutput {\n  /**\n   * The bundling output directory includes a single .zip or .jar file which\n   * will be used as the final bundle. If the output directory does not\n   * include exactly a single archive, bundling will fail.\n   */\n  ARCHIVED = 'archived',\n\n  /**\n   * The bundling output directory contains one or more files which will be\n   * archived and uploaded as a .zip file to S3.\n   */\n  NOT_ARCHIVED = 'not-archived',\n\n  /**\n   * If the bundling output directory contains a single archive file (zip or jar)\n   * it will be used as the bundle output as-is. Otherwise all the files in the bundling output directory will be zipped.\n   */\n  AUTO_DISCOVER = 'auto-discover',\n}\n\n/**\n * Local bundling\n *\n */\nexport interface ILocalBundling {\n  /**\n   * This method is called before attempting docker bundling to allow the\n   * bundler to be executed locally. If the local bundler exists, and bundling\n   * was performed locally, return `true`. Otherwise, return `false`.\n   *\n   * @param outputDir the directory where the bundled asset should be output\n   * @param options bundling options for this asset\n   */\n  tryBundle(outputDir: string, options: BundlingOptions): boolean;\n}\n\n/**\n * A Docker image used for asset bundling\n *\n * @deprecated use DockerImage\n */\nexport class BundlingDockerImage {\n  /**\n   * Reference an image on DockerHub or another online registry.\n   *\n   * @param image the image name\n   */\n  public static fromRegistry(image: string) {\n    return new DockerImage(image);\n  }\n\n  /**\n   * Reference an image that's built directly from sources on disk.\n   *\n   * @param path The path to the directory containing the Docker file\n   * @param options Docker build options\n   *\n   * @deprecated use DockerImage.fromBuild()\n   */\n  public static fromAsset(path: string, options: DockerBuildOptions = {}): BundlingDockerImage {\n    return DockerImage.fromBuild(path, options);\n  }\n\n  /** @param image The Docker image */\n  protected constructor(public readonly image: string, private readonly _imageHash?: string) {}\n\n  /**\n   * Provides a stable representation of this image for JSON serialization.\n   *\n   * @return The overridden image name if set or image hash name in that order\n   */\n  public toJSON() {\n    return this._imageHash ?? this.image;\n  }\n\n  /**\n   * Runs a Docker image\n   */\n  public run(options: DockerRunOptions = {}) {\n    const volumes = options.volumes || [];\n    const environment = options.environment || {};\n    const entrypoint = options.entrypoint?.[0] || null;\n    const command = [\n      ...options.entrypoint?.[1]\n        ? [...options.entrypoint.slice(1)]\n        : [],\n      ...options.command\n        ? [...options.command]\n        : [],\n    ];\n\n    const dockerArgs: string[] = [\n      'run', '--rm',\n      ...options.securityOpt\n        ? ['--security-opt', options.securityOpt]\n        : [],\n      ...options.user\n        ? ['-u', options.user]\n        : [],\n      ...flatten(volumes.map(v => ['-v', `${v.hostPath}:${v.containerPath}:${isSeLinux() ? 'z,' : ''}${v.consistency ?? DockerVolumeConsistency.DELEGATED}`])),\n      ...flatten(Object.entries(environment).map(([k, v]) => ['--env', `${k}=${v}`])),\n      ...options.workingDirectory\n        ? ['-w', options.workingDirectory]\n        : [],\n      ...entrypoint\n        ? ['--entrypoint', entrypoint]\n        : [],\n      this.image,\n      ...command,\n    ];\n\n    dockerExec(dockerArgs);\n  }\n\n  /**\n   * Copies a file or directory out of the Docker image to the local filesystem.\n   *\n   * If `outputPath` is omitted the destination path is a temporary directory.\n   *\n   * @param imagePath the path in the Docker image\n   * @param outputPath the destination path for the copy operation\n   * @returns the destination path\n   */\n  public cp(imagePath: string, outputPath?: string): string {\n    const { stdout } = dockerExec(['create', this.image], {}); // Empty options to avoid stdout redirect here\n    const match = stdout.toString().match(/([0-9a-f]{16,})/);\n    if (!match) {\n      throw new Error('Failed to extract container ID from Docker create output');\n    }\n\n    const containerId = match[1];\n    const containerPath = `${containerId}:${imagePath}`;\n    const destPath = outputPath ?? FileSystem.mkdtemp('cdk-docker-cp-');\n    try {\n      dockerExec(['cp', containerPath, destPath]);\n      return destPath;\n    } catch (err) {\n      throw new Error(`Failed to copy files from ${containerPath} to ${destPath}: ${err}`);\n    } finally {\n      dockerExec(['rm', '-v', containerId]);\n    }\n  }\n}\n\n/**\n * A Docker image\n */\nexport class DockerImage extends BundlingDockerImage {\n  /**\n   * Builds a Docker image\n   *\n   * @param path The path to the directory containing the Docker file\n   * @param options Docker build options\n   */\n  public static fromBuild(path: string, options: DockerBuildOptions = {}) {\n    const buildArgs = options.buildArgs || {};\n\n    if (options.file && isAbsolute(options.file)) {\n      throw new Error(`\"file\" must be relative to the docker build directory. Got ${options.file}`);\n    }\n\n    // Image tag derived from path and build options\n    const input = JSON.stringify({ path, ...options });\n    const tagHash = crypto.createHash('sha256').update(input).digest('hex');\n    const tag = `cdk-${tagHash}`;\n\n    const dockerArgs: string[] = [\n      'build', '-t', tag,\n      ...(options.file ? ['-f', join(path, options.file)] : []),\n      ...(options.platform ? ['--platform', options.platform] : []),\n      ...flatten(Object.entries(buildArgs).map(([k, v]) => ['--build-arg', `${k}=${v}`])),\n      path,\n    ];\n\n    dockerExec(dockerArgs);\n\n    // Fingerprints the directory containing the Dockerfile we're building and\n    // differentiates the fingerprint based on build arguments. We do this so\n    // we can provide a stable image hash. Otherwise, the image ID will be\n    // different every time the Docker layer cache is cleared, due primarily to\n    // timestamps.\n    const hash = FileSystem.fingerprint(path, { extraHash: JSON.stringify(options) });\n    return new DockerImage(tag, hash);\n  }\n\n  /**\n   * Reference an image on DockerHub or another online registry.\n   *\n   * @param image the image name\n   */\n  public static fromRegistry(image: string) {\n    return new DockerImage(image);\n  }\n\n  /** The Docker image */\n  public readonly image: string;\n\n  constructor(image: string, _imageHash?: string) {\n    // It is preferrable for the deprecated class to inherit a non-deprecated class.\n    // However, in this case, the opposite has occurred which is incompatible with\n    // a deprecation feature. See https://github.com/aws/jsii/issues/3102.\n    const deprecated = quiet();\n\n    super(image, _imageHash);\n\n    reset(deprecated);\n    this.image = image;\n  }\n\n  /**\n   * Provides a stable representation of this image for JSON serialization.\n   *\n   * @return The overridden image name if set or image hash name in that order\n   */\n  public toJSON() {\n    // It is preferrable for the deprecated class to inherit a non-deprecated class.\n    // However, in this case, the opposite has occurred which is incompatible with\n    // a deprecation feature. See https://github.com/aws/jsii/issues/3102.\n    const deprecated = quiet();\n\n    const json = super.toJSON();\n\n    reset(deprecated);\n    return json;\n  }\n\n  /**\n   * Runs a Docker image\n   */\n  public run(options: DockerRunOptions = {}) {\n    // It is preferrable for the deprecated class to inherit a non-deprecated class.\n    // However, in this case, the opposite has occurred which is incompatible with\n    // a deprecation feature. See https://github.com/aws/jsii/issues/3102.\n    const deprecated = quiet();\n\n    const result = super.run(options);\n\n    reset(deprecated);\n    return result;\n  }\n\n  /**\n   * Copies a file or directory out of the Docker image to the local filesystem.\n   *\n   * If `outputPath` is omitted the destination path is a temporary directory.\n   *\n   * @param imagePath the path in the Docker image\n   * @param outputPath the destination path for the copy operation\n   * @returns the destination path\n   */\n  public cp(imagePath: string, outputPath?: string): string {\n    // It is preferrable for the deprecated class to inherit a non-deprecated class.\n    // However, in this case, the opposite has occurred which is incompatible with\n    // a deprecation feature. See https://github.com/aws/jsii/issues/3102.\n    const deprecated = quiet();\n\n    const result = super.cp(imagePath, outputPath);\n\n    reset(deprecated);\n    return result;\n  }\n}\n\n/**\n * A Docker volume\n */\nexport interface DockerVolume {\n  /**\n   * The path to the file or directory on the host machine\n   */\n  readonly hostPath: string;\n\n  /**\n   * The path where the file or directory is mounted in the container\n   */\n  readonly containerPath: string;\n\n  /**\n   * Mount consistency. Only applicable for macOS\n   *\n   * @default DockerConsistency.DELEGATED\n   * @see https://docs.docker.com/storage/bind-mounts/#configure-mount-consistency-for-macos\n   */\n  readonly consistency?: DockerVolumeConsistency;\n}\n\n/**\n * Supported Docker volume consistency types. Only valid on macOS due to the way file storage works on Mac\n */\nexport enum DockerVolumeConsistency {\n  /**\n   * Read/write operations inside the Docker container are applied immediately on the mounted host machine volumes\n   */\n  CONSISTENT = 'consistent',\n  /**\n   * Read/write operations on mounted Docker volumes are first written inside the container and then synchronized to the host machine\n   */\n  DELEGATED = 'delegated',\n  /**\n   * Read/write operations on mounted Docker volumes are first applied on the host machine and then synchronized to the container\n   */\n  CACHED = 'cached',\n}\n\n/**\n * Docker run options\n */\nexport interface DockerRunOptions {\n  /**\n   * The entrypoint to run in the container.\n   *\n   * @default - run the entrypoint defined in the image\n   */\n  readonly entrypoint?: string[];\n\n  /**\n   * The command to run in the container.\n   *\n   * @default - run the command defined in the image\n   */\n  readonly command?: string[];\n\n  /**\n   * Docker volumes to mount.\n   *\n   * @default - no volumes are mounted\n   */\n  readonly volumes?: DockerVolume[];\n\n  /**\n   * The environment variables to pass to the container.\n   *\n   * @default - no environment variables.\n   */\n  readonly environment?: { [key: string]: string; };\n\n  /**\n   * Working directory inside the container.\n   *\n   * @default - image default\n   */\n  readonly workingDirectory?: string;\n\n  /**\n   * The user to use when running the container.\n   *\n   * @default - root or image default\n   */\n  readonly user?: string;\n\n  /**\n   * [Security configuration](https://docs.docker.com/engine/reference/run/#security-configuration)\n   * when running the docker container.\n   *\n   * @default - no security options\n   */\n  readonly securityOpt?: string;\n}\n\n/**\n * Docker build options\n */\nexport interface DockerBuildOptions {\n  /**\n   * Build args\n   *\n   * @default - no build args\n   */\n  readonly buildArgs?: { [key: string]: string };\n\n  /**\n   * Name of the Dockerfile, must relative to the docker build path.\n   *\n   * @default `Dockerfile`\n   */\n  readonly file?: string;\n\n  /**\n   * Set platform if server is multi-platform capable. _Requires Docker Engine API v1.38+_.\n   *\n   * Example value: `linux/amd64`\n   *\n   * @default - no platform specified\n   */\n  readonly platform?: string;\n}\n\nfunction flatten(x: string[][]) {\n  return Array.prototype.concat([], ...x);\n}\n\nfunction dockerExec(args: string[], options?: SpawnSyncOptions) {\n  const prog = process.env.CDK_DOCKER ?? 'docker';\n  const proc = spawnSync(prog, args, options ?? {\n    stdio: [ // show Docker output\n      'ignore', // ignore stdio\n      process.stderr, // redirect stdout to stderr\n      'inherit', // inherit stderr\n    ],\n  });\n\n  if (proc.error) {\n    throw proc.error;\n  }\n\n  if (proc.status !== 0) {\n    if (proc.stdout || proc.stderr) {\n      throw new Error(`[Status ${proc.status}] stdout: ${proc.stdout?.toString().trim()}\\n\\n\\nstderr: ${proc.stderr?.toString().trim()}`);\n    }\n    throw new Error(`${prog} exited with status ${proc.status}`);\n  }\n\n  return proc;\n}\n\nfunction isSeLinux() : boolean {\n  if (process.platform != 'linux') {\n    return false;\n  }\n  const prog = 'selinuxenabled';\n  const proc = spawnSync(prog, [], {\n    stdio: [ // show selinux status output\n      'pipe', // get value of stdio\n      process.stderr, // redirect stdout to stderr\n      'inherit', // inherit stderr\n    ],\n  });\n  if (proc.error) {\n    // selinuxenabled not a valid command, therefore not enabled\n    return false;\n  }\n  if (proc.status == 0) {\n    // selinux enabled\n    return true;\n  } else {\n    // selinux not enabled\n    return false;\n  }\n}\n"]}
\No newline at end of file