cdk-assets 2.63.2 → 2.65.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -10,7 +10,7 @@ const archiver = require('archiver');
10
10
  async function zipDirectory(directory, outputFile, logger) {
11
11
  // We write to a temporary file and rename at the last moment. This is so that if we are
12
12
  // interrupted during this process, we don't leave a half-finished file in the target location.
13
- const temporaryOutputFile = `${outputFile}._tmp`;
13
+ const temporaryOutputFile = `${outputFile}.${randomString()}._tmp`;
14
14
  await writeZipFile(directory, temporaryOutputFile);
15
15
  await moveIntoPlace(temporaryOutputFile, outputFile, logger);
16
16
  }
@@ -89,4 +89,7 @@ async function pathExists(x) {
89
89
  throw e;
90
90
  }
91
91
  }
92
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXJjaGl2ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImFyY2hpdmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsMkJBQXVEO0FBQ3ZELDZCQUE2QjtBQUM3Qiw2QkFBNkI7QUFFN0IseUVBQXlFO0FBQ3pFLGlFQUFpRTtBQUNqRSxNQUFNLFFBQVEsR0FBRyxPQUFPLENBQUMsVUFBVSxDQUFDLENBQUM7QUFJOUIsS0FBSyxVQUFVLFlBQVksQ0FBQyxTQUFpQixFQUFFLFVBQWtCLEVBQUUsTUFBYztJQUN0Rix3RkFBd0Y7SUFDeEYsK0ZBQStGO0lBQy9GLE1BQU0sbUJBQW1CLEdBQUcsR0FBRyxVQUFVLE9BQU8sQ0FBQztJQUNqRCxNQUFNLFlBQVksQ0FBQyxTQUFTLEVBQUUsbUJBQW1CLENBQUMsQ0FBQztJQUNuRCxNQUFNLGFBQWEsQ0FBQyxtQkFBbUIsRUFBRSxVQUFVLEVBQUUsTUFBTSxDQUFDLENBQUM7QUFDL0QsQ0FBQztBQU5ELG9DQU1DO0FBRUQsU0FBUyxZQUFZLENBQUMsU0FBaUIsRUFBRSxVQUFrQjtJQUN6RCxPQUFPLElBQUksT0FBTyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsSUFBSSxFQUFFLEVBQUU7UUFDcEMsc0ZBQXNGO1FBQ3RGLGlGQUFpRjtRQUNqRixpRUFBaUU7UUFDakUsTUFBTSxXQUFXLEdBQUc7WUFDbEIsR0FBRyxFQUFFLElBQUk7WUFDVCxLQUFLLEVBQUUsSUFBSTtZQUNYLE1BQU0sRUFBRSxJQUFJO1lBQ1osR0FBRyxFQUFFLFNBQVM7U0FDZixDQUFDO1FBQ0YsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsV0FBVyxDQUFDLENBQUMsQ0FBQyxvQ0FBb0M7UUFFaEYsTUFBTSxNQUFNLEdBQUcsc0JBQWlCLENBQUMsVUFBVSxDQUFDLENBQUM7UUFFN0MsTUFBTSxPQUFPLEdBQUcsUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ2hDLE9BQU8sQ0FBQyxFQUFFLENBQUMsU0FBUyxFQUFFLElBQUksQ0FBQyxDQUFDO1FBQzVCLE9BQU8sQ0FBQyxFQUFFLENBQUMsT0FBTyxFQUFFLElBQUksQ0FBQyxDQUFDO1FBRTFCLHdGQUF3RjtRQUN4Riw2RkFBNkY7UUFDN0YsNkNBQTZDO1FBQzdDLE1BQU0sQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBRXpCLE9BQU8sQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7UUFFckIsNkNBQTZDO1FBQzdDLEtBQUssTUFBTSxJQUFJLElBQUksS0FBSyxFQUFFO1lBQ3hCLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsU0FBUyxFQUFFLElBQUksQ0FBQyxDQUFDO1lBQy9DLE1BQU0sQ0FBQyxJQUFJLEVBQUUsSUFBSSxDQUFDLEdBQUcsTUFBTSxPQUFPLENBQUMsR0FBRyxDQUFDLENBQUMsYUFBRSxDQUFDLFFBQVEsQ0FBQyxRQUFRLENBQUMsRUFBRSxhQUFFLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUNuRixPQUFPLENBQUMsTUFBTSxDQUFDLElBQUksRUFBRTtnQkFDbkIsSUFBSSxFQUFFLElBQUk7Z0JBQ1YsSUFBSSxFQUFFLElBQUksSUFBSSxDQUFDLDBCQUEwQixDQUFDO2dCQUMxQyxJQUFJLEVBQUUsSUFBSSxDQUFDLElBQUk7YUFDaEIsQ0FBQyxDQUFDO1NBQ0o7UUFFRCxNQUFNLE9BQU8sQ0FBQyxRQUFRLEVBQUUsQ0FBQztJQUMzQixDQUFDLENBQUMsQ0FBQztBQUNMLENBQUM7QUFFRDs7O0dBR0c7QUFDSCxLQUFLLFVBQVUsYUFBYSxDQUFDLE1BQWMsRUFBRSxNQUFjLEVBQUUsTUFBYztJQUN6RSxJQUFJLEtBQUssR0FBRyxHQUFHLENBQUM7SUFDaEIsSUFBSSxRQUFRLEdBQUcsQ0FBQyxDQUFDO0lBQ2pCLE9BQU8sSUFBSSxFQUFFO1FBQ1gsSUFBSTtZQUNGLElBQUksTUFBTSxVQUFVLENBQUMsTUFBTSxDQUFDLEVBQUU7Z0JBQzVCLE1BQU0sYUFBRSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQzthQUN6QjtZQUNELE1BQU0sYUFBRSxDQUFDLE1BQU0sQ0FBQyxNQUFNLEVBQUUsTUFBTSxDQUFDLENBQUM7WUFDaEMsT0FBTztTQUNSO1FBQUMsT0FBTyxDQUFDLEVBQUU7WUFDVixJQUFJLENBQUMsQ0FBQyxJQUFJLEtBQUssT0FBTyxJQUFJLFFBQVEsRUFBRSxJQUFJLENBQUMsRUFBRTtnQkFDekMsTUFBTSxDQUFDLENBQUM7YUFDVDtZQUNELE1BQU0sQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUM7WUFDbEIsTUFBTSxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLEdBQUcsS0FBSyxDQUFDLENBQUMsQ0FBQztZQUMvQyxLQUFLLElBQUksQ0FBQyxDQUFDO1NBQ1o7S0FDRjtBQUNILENBQUM7QUFFRCxTQUFTLEtBQUssQ0FBQyxFQUFVO0lBQ3ZCLE9BQU8sSUFBSSxPQUFPLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxVQUFVLENBQUMsRUFBRSxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUM7QUFDL0MsQ0FBQztBQUVELEtBQUssVUFBVSxVQUFVLENBQUMsQ0FBUztJQUNqQyxJQUFJO1FBQ0YsTUFBTSxhQUFFLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2pCLE9BQU8sSUFBSSxDQUFDO0tBQ2I7SUFBQyxPQUFPLENBQUMsRUFBRTtRQUNWLElBQUksQ0FBQyxDQUFDLElBQUksS0FBSyxRQUFRLEVBQUU7WUFDdkIsT0FBTyxLQUFLLENBQUM7U0FDZDtRQUNELE1BQU0sQ0FBQyxDQUFDO0tBQ1Q7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3JlYXRlV3JpdGVTdHJlYW0sIHByb21pc2VzIGFzIGZzIH0gZnJvbSAnZnMnO1xuaW1wb3J0ICogYXMgcGF0aCBmcm9tICdwYXRoJztcbmltcG9ydCAqIGFzIGdsb2IgZnJvbSAnZ2xvYic7XG5cbi8vIG5hbWVzcGFjZSBvYmplY3QgaW1wb3J0cyB3b24ndCB3b3JrIGluIHRoZSBidW5kbGUgZm9yIGZ1bmN0aW9uIGV4cG9ydHNcbi8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzXG5jb25zdCBhcmNoaXZlciA9IHJlcXVpcmUoJ2FyY2hpdmVyJyk7XG5cbnR5cGUgTG9nZ2VyID0gKHg6IHN0cmluZykgPT4gdm9pZDtcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHppcERpcmVjdG9yeShkaXJlY3Rvcnk6IHN0cmluZywgb3V0cHV0RmlsZTogc3RyaW5nLCBsb2dnZXI6IExvZ2dlcik6IFByb21pc2U8dm9pZD4ge1xuICAvLyBXZSB3cml0ZSB0byBhIHRlbXBvcmFyeSBmaWxlIGFuZCByZW5hbWUgYXQgdGhlIGxhc3QgbW9tZW50LiBUaGlzIGlzIHNvIHRoYXQgaWYgd2UgYXJlXG4gIC8vIGludGVycnVwdGVkIGR1cmluZyB0aGlzIHByb2Nlc3MsIHdlIGRvbid0IGxlYXZlIGEgaGFsZi1maW5pc2hlZCBmaWxlIGluIHRoZSB0YXJnZXQgbG9jYXRpb24uXG4gIGNvbnN0IHRlbXBvcmFyeU91dHB1dEZpbGUgPSBgJHtvdXRwdXRGaWxlfS5fdG1wYDtcbiAgYXdhaXQgd3JpdGVaaXBGaWxlKGRpcmVjdG9yeSwgdGVtcG9yYXJ5T3V0cHV0RmlsZSk7XG4gIGF3YWl0IG1vdmVJbnRvUGxhY2UodGVtcG9yYXJ5T3V0cHV0RmlsZSwgb3V0cHV0RmlsZSwgbG9nZ2VyKTtcbn1cblxuZnVuY3Rpb24gd3JpdGVaaXBGaWxlKGRpcmVjdG9yeTogc3RyaW5nLCBvdXRwdXRGaWxlOiBzdHJpbmcpOiBQcm9taXNlPHZvaWQ+IHtcbiAgcmV0dXJuIG5ldyBQcm9taXNlKGFzeW5jIChvaywgZmFpbCkgPT4ge1xuICAgIC8vIFRoZSBiZWxvdyBvcHRpb25zIGFyZSBuZWVkZWQgdG8gc3VwcG9ydCBmb2xsb3dpbmcgc3ltbGlua3Mgd2hlbiBidWlsZGluZyB6aXAgZmlsZXM6XG4gICAgLy8gLSBub2RpcjogVGhpcyB3aWxsIHByZXZlbnQgc3ltbGlua3MgdGhlbXNlbHZlcyBmcm9tIGJlaW5nIGNvcGllZCBpbnRvIHRoZSB6aXAuXG4gICAgLy8gLSBmb2xsb3c6IFRoaXMgd2lsbCBmb2xsb3cgc3ltbGlua3MgYW5kIGNvcHkgdGhlIGZpbGVzIHdpdGhpbi5cbiAgICBjb25zdCBnbG9iT3B0aW9ucyA9IHtcbiAgICAgIGRvdDogdHJ1ZSxcbiAgICAgIG5vZGlyOiB0cnVlLFxuICAgICAgZm9sbG93OiB0cnVlLFxuICAgICAgY3dkOiBkaXJlY3RvcnksXG4gICAgfTtcbiAgICBjb25zdCBmaWxlcyA9IGdsb2Iuc3luYygnKionLCBnbG9iT3B0aW9ucyk7IC8vIFRoZSBvdXRwdXQgaGVyZSBpcyBhbHJlYWR5IHNvcnRlZFxuXG4gICAgY29uc3Qgb3V0cHV0ID0gY3JlYXRlV3JpdGVTdHJlYW0ob3V0cHV0RmlsZSk7XG5cbiAgICBjb25zdCBhcmNoaXZlID0gYXJjaGl2ZXIoJ3ppcCcpO1xuICAgIGFyY2hpdmUub24oJ3dhcm5pbmcnLCBmYWlsKTtcbiAgICBhcmNoaXZlLm9uKCdlcnJvcicsIGZhaWwpO1xuXG4gICAgLy8gYXJjaGl2ZSBoYXMgYmVlbiBmaW5hbGl6ZWQgYW5kIHRoZSBvdXRwdXQgZmlsZSBkZXNjcmlwdG9yIGhhcyBjbG9zZWQsIHJlc29sdmUgcHJvbWlzZVxuICAgIC8vIHRoaXMgaGFzIHRvIGJlIGRvbmUgYmVmb3JlIGNhbGxpbmcgYGZpbmFsaXplYCBzaW5jZSB0aGUgZXZlbnRzIG1heSBmaXJlIGltbWVkaWF0ZWx5IGFmdGVyLlxuICAgIC8vIHNlZSBodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9hcmNoaXZlclxuICAgIG91dHB1dC5vbmNlKCdjbG9zZScsIG9rKTtcblxuICAgIGFyY2hpdmUucGlwZShvdXRwdXQpO1xuXG4gICAgLy8gQXBwZW5kIGZpbGVzIHNlcmlhbGx5IHRvIGVuc3VyZSBmaWxlIG9yZGVyXG4gICAgZm9yIChjb25zdCBmaWxlIG9mIGZpbGVzKSB7XG4gICAgICBjb25zdCBmdWxsUGF0aCA9IHBhdGgucmVzb2x2ZShkaXJlY3RvcnksIGZpbGUpO1xuICAgICAgY29uc3QgW2RhdGEsIHN0YXRdID0gYXdhaXQgUHJvbWlzZS5hbGwoW2ZzLnJlYWRGaWxlKGZ1bGxQYXRoKSwgZnMuc3RhdChmdWxsUGF0aCldKTtcbiAgICAgIGFyY2hpdmUuYXBwZW5kKGRhdGEsIHtcbiAgICAgICAgbmFtZTogZmlsZSxcbiAgICAgICAgZGF0ZTogbmV3IERhdGUoJzE5ODAtMDEtMDFUMDA6MDA6MDAuMDAwWicpLCAvLyByZXNldCBkYXRlcyB0byBnZXQgdGhlIHNhbWUgaGFzaCBmb3IgdGhlIHNhbWUgY29udGVudFxuICAgICAgICBtb2RlOiBzdGF0Lm1vZGUsXG4gICAgICB9KTtcbiAgICB9XG5cbiAgICBhd2FpdCBhcmNoaXZlLmZpbmFsaXplKCk7XG4gIH0pO1xufVxuXG4vKipcbiAqIFJlbmFtZSB0aGUgZmlsZSB0byB0aGUgdGFyZ2V0IGxvY2F0aW9uLCB0YWtpbmcgaW50byBhY2NvdW50IHRoYXQgd2UgbWF5IHNlZSBFUEVSTSBvbiBXaW5kb3dzXG4gKiB3aGlsZSBhbiBBbnRpdmlydXMgc2Nhbm5lciBzdGlsbCBoYXMgdGhlIGZpbGUgb3Blbiwgc28gcmV0cnkgYSBjb3VwbGUgb2YgdGltZXMuXG4gKi9cbmFzeW5jIGZ1bmN0aW9uIG1vdmVJbnRvUGxhY2Uoc291cmNlOiBzdHJpbmcsIHRhcmdldDogc3RyaW5nLCBsb2dnZXI6IExvZ2dlcikge1xuICBsZXQgZGVsYXkgPSAxMDA7XG4gIGxldCBhdHRlbXB0cyA9IDU7XG4gIHdoaWxlICh0cnVlKSB7XG4gICAgdHJ5IHtcbiAgICAgIGlmIChhd2FpdCBwYXRoRXhpc3RzKHRhcmdldCkpIHtcbiAgICAgICAgYXdhaXQgZnMudW5saW5rKHRhcmdldCk7XG4gICAgICB9XG4gICAgICBhd2FpdCBmcy5yZW5hbWUoc291cmNlLCB0YXJnZXQpO1xuICAgICAgcmV0dXJuO1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIGlmIChlLmNvZGUgIT09ICdFUEVSTScgfHwgYXR0ZW1wdHMtLSA8PSAwKSB7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgICBsb2dnZXIoZS5tZXNzYWdlKTtcbiAgICAgIGF3YWl0IHNsZWVwKE1hdGguZmxvb3IoTWF0aC5yYW5kb20oKSAqIGRlbGF5KSk7XG4gICAgICBkZWxheSAqPSAyO1xuICAgIH1cbiAgfVxufVxuXG5mdW5jdGlvbiBzbGVlcChtczogbnVtYmVyKSB7XG4gIHJldHVybiBuZXcgUHJvbWlzZShvayA9PiBzZXRUaW1lb3V0KG9rLCBtcykpO1xufVxuXG5hc3luYyBmdW5jdGlvbiBwYXRoRXhpc3RzKHg6IHN0cmluZykge1xuICB0cnkge1xuICAgIGF3YWl0IGZzLnN0YXQoeCk7XG4gICAgcmV0dXJuIHRydWU7XG4gIH0gY2F0Y2ggKGUpIHtcbiAgICBpZiAoZS5jb2RlID09PSAnRU5PRU5UJykge1xuICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgICB0aHJvdyBlO1xuICB9XG59Il19
92
+ function randomString() {
93
+ return Math.random().toString(36).replace(/[^a-z0-9]+/g, '');
94
+ }
95
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"archive.js","sourceRoot":"","sources":["archive.ts"],"names":[],"mappings":"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","sourcesContent":["import { createWriteStream, promises as fs } from 'fs';\nimport * as path from 'path';\nimport * as glob from 'glob';\n\n// namespace object imports won't work in the bundle for function exports\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst archiver = require('archiver');\n\ntype Logger = (x: string) => void;\n\nexport async function zipDirectory(directory: string, outputFile: string, logger: Logger): Promise<void> {\n  // We write to a temporary file and rename at the last moment. This is so that if we are\n  // interrupted during this process, we don't leave a half-finished file in the target location.\n  const temporaryOutputFile = `${outputFile}.${randomString()}._tmp`;\n  await writeZipFile(directory, temporaryOutputFile);\n  await moveIntoPlace(temporaryOutputFile, outputFile, logger);\n}\n\nfunction writeZipFile(directory: string, outputFile: string): Promise<void> {\n  return new Promise(async (ok, fail) => {\n    // The below options are needed to support following symlinks when building zip files:\n    // - nodir: This will prevent symlinks themselves from being copied into the zip.\n    // - follow: This will follow symlinks and copy the files within.\n    const globOptions = {\n      dot: true,\n      nodir: true,\n      follow: true,\n      cwd: directory,\n    };\n    const files = glob.sync('**', globOptions); // The output here is already sorted\n\n    const output = createWriteStream(outputFile);\n\n    const archive = archiver('zip');\n    archive.on('warning', fail);\n    archive.on('error', fail);\n\n    // archive has been finalized and the output file descriptor has closed, resolve promise\n    // this has to be done before calling `finalize` since the events may fire immediately after.\n    // see https://www.npmjs.com/package/archiver\n    output.once('close', ok);\n\n    archive.pipe(output);\n\n    // Append files serially to ensure file order\n    for (const file of files) {\n      const fullPath = path.resolve(directory, file);\n      const [data, stat] = await Promise.all([fs.readFile(fullPath), fs.stat(fullPath)]);\n      archive.append(data, {\n        name: file,\n        date: new Date('1980-01-01T00:00:00.000Z'), // reset dates to get the same hash for the same content\n        mode: stat.mode,\n      });\n    }\n\n    await archive.finalize();\n  });\n}\n\n/**\n * Rename the file to the target location, taking into account that we may see EPERM on Windows\n * while an Antivirus scanner still has the file open, so retry a couple of times.\n */\nasync function moveIntoPlace(source: string, target: string, logger: Logger) {\n  let delay = 100;\n  let attempts = 5;\n  while (true) {\n    try {\n      if (await pathExists(target)) {\n        await fs.unlink(target);\n      }\n      await fs.rename(source, target);\n      return;\n    } catch (e) {\n      if (e.code !== 'EPERM' || attempts-- <= 0) {\n        throw e;\n      }\n      logger(e.message);\n      await sleep(Math.floor(Math.random() * delay));\n      delay *= 2;\n    }\n  }\n}\n\nfunction sleep(ms: number) {\n  return new Promise(ok => setTimeout(ok, ms));\n}\n\nasync function pathExists(x: string) {\n  try {\n    await fs.stat(x);\n    return true;\n  } catch (e) {\n    if (e.code === 'ENOENT') {\n      return false;\n    }\n    throw e;\n  }\n}\n\nfunction randomString() {\n  return Math.random().toString(36).replace(/[^a-z0-9]+/g, '');\n}"]}
@@ -8,6 +8,7 @@ interface BuildOptions {
8
8
  readonly target?: string;
9
9
  readonly file?: string;
10
10
  readonly buildArgs?: Record<string, string>;
11
+ readonly buildSecrets?: Record<string, string>;
11
12
  readonly networkMode?: string;
12
13
  readonly platform?: string;
13
14
  readonly outputs?: string[];
@@ -31,6 +31,7 @@ class Docker {
31
31
  const buildCommand = [
32
32
  'build',
33
33
  ...flatten(Object.entries(options.buildArgs || {}).map(([k, v]) => ['--build-arg', `${k}=${v}`])),
34
+ ...flatten(Object.entries(options.buildSecrets || {}).map(([k, v]) => ['--secret', `id=${k},${v}`])),
34
35
  '--tag', options.tag,
35
36
  ...options.target ? ['--target', options.target] : [],
36
37
  ...options.file ? ['--file', options.file] : [],
@@ -173,4 +174,4 @@ function getDockerCmd() {
173
174
  function flatten(x) {
174
175
  return Array.prototype.concat([], ...x);
175
176
  }
176
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docker.js","sourceRoot":"","sources":["docker.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { cdkCredentialsConfig, obtainEcrCredentials } from './docker-credentials';\nimport { Logger, shell, ShellOptions } from './shell';\nimport { createCriticalSection } from './util';\n\ninterface BuildOptions {\n  readonly directory: string;\n\n  /**\n   * Tag the image with a given repoName:tag combination\n   */\n  readonly tag: string;\n  readonly target?: string;\n  readonly file?: string;\n  readonly buildArgs?: Record<string, string>;\n  readonly networkMode?: string;\n  readonly platform?: string;\n  readonly outputs?: string[];\n}\n\nexport interface DockerCredentialsConfig {\n  readonly version: string;\n  readonly domainCredentials: Record<string, DockerDomainCredentials>;\n}\n\nexport interface DockerDomainCredentials {\n  readonly secretsManagerSecretId?: string;\n  readonly ecrRepository?: string;\n}\n\nexport class Docker {\n\n  private configDir: string | undefined = undefined;\n\n  constructor(private readonly logger?: Logger) {\n  }\n\n  /**\n   * Whether an image with the given tag exists\n   */\n  public async exists(tag: string) {\n    try {\n      await this.execute(['inspect', tag], { quiet: true });\n      return true;\n    } catch (e) {\n      if (e.code !== 'PROCESS_FAILED' || e.exitCode !== 1) { throw e; }\n      return false;\n    }\n  }\n\n  public async build(options: BuildOptions) {\n    const buildCommand = [\n      'build',\n      ...flatten(Object.entries(options.buildArgs || {}).map(([k, v]) => ['--build-arg', `${k}=${v}`])),\n      '--tag', options.tag,\n      ...options.target ? ['--target', options.target] : [],\n      ...options.file ? ['--file', options.file] : [],\n      ...options.networkMode ? ['--network', options.networkMode] : [],\n      ...options.platform ? ['--platform', options.platform] : [],\n      ...options.outputs ? options.outputs.map(output => [`--output=${output}`]) : [],\n      '.',\n    ];\n    await this.execute(buildCommand, { cwd: options.directory });\n  }\n\n  /**\n   * Get credentials from ECR and run docker login\n   */\n  public async login(ecr: AWS.ECR) {\n    const credentials = await obtainEcrCredentials(ecr);\n\n    // Use --password-stdin otherwise docker will complain. Loudly.\n    await this.execute(['login',\n      '--username', credentials.username,\n      '--password-stdin',\n      credentials.endpoint], {\n      input: credentials.password,\n\n      // Need to quiet otherwise Docker will complain\n      // 'WARNING! Your password will be stored unencrypted'\n      // doesn't really matter since it's a token.\n      quiet: true,\n    });\n  }\n\n  public async tag(sourceTag: string, targetTag: string) {\n    await this.execute(['tag', sourceTag, targetTag]);\n  }\n\n  public async push(tag: string) {\n    await this.execute(['push', tag]);\n  }\n\n  /**\n   * If a CDK Docker Credentials file exists, creates a new Docker config directory.\n   * Sets up `docker-credential-cdk-assets` to be the credential helper for each domain in the CDK config.\n   * All future commands (e.g., `build`, `push`) will use this config.\n   *\n   * See https://docs.docker.com/engine/reference/commandline/login/#credential-helpers for more details on cred helpers.\n   *\n   * @returns true if CDK config was found and configured, false otherwise\n   */\n  public configureCdkCredentials(): boolean {\n    const config = cdkCredentialsConfig();\n    if (!config) { return false; }\n\n    this.configDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdkDockerConfig'));\n\n    const domains = Object.keys(config.domainCredentials);\n    const credHelpers = domains.reduce((map: Record<string, string>, domain) => {\n      map[domain] = 'cdk-assets'; // Use docker-credential-cdk-assets for this domain\n      return map;\n    }, {});\n    fs.writeFileSync(path.join(this.configDir, 'config.json'), JSON.stringify({ credHelpers }), { encoding: 'utf-8' });\n\n    return true;\n  }\n\n  /**\n   * Removes any configured Docker config directory.\n   * All future commands (e.g., `build`, `push`) will use the default config.\n   *\n   * This is useful after calling `configureCdkCredentials` to reset to default credentials.\n   */\n  public resetAuthPlugins() {\n    this.configDir = undefined;\n  }\n\n  private async execute(args: string[], options: ShellOptions = {}) {\n    const configArgs = this.configDir ? ['--config', this.configDir] : [];\n\n    const pathToCdkAssets = path.resolve(__dirname, '..', '..', 'bin');\n    try {\n      await shell([getDockerCmd(), ...configArgs, ...args], {\n        logger: this.logger,\n        ...options,\n        env: {\n          ...process.env,\n          ...options.env,\n          PATH: `${pathToCdkAssets}${path.delimiter}${options.env?.PATH ?? process.env.PATH}`,\n        },\n      });\n    } catch (e) {\n      if (e.code === 'ENOENT') {\n        throw new Error('Unable to execute \\'docker\\' in order to build a container asset. Please install \\'docker\\' and try again.');\n      }\n      throw e;\n    }\n  }\n}\n\nexport interface DockerFactoryOptions {\n  readonly repoUri: string;\n  readonly ecr: AWS.ECR;\n  readonly logger: (m: string) => void;\n}\n\n/**\n * Helps get appropriately configured Docker instances during the container\n * image publishing process.\n */\nexport class DockerFactory {\n  private enterLoggedInDestinationsCriticalSection = createCriticalSection();\n  private loggedInDestinations = new Set<string>();\n\n  /**\n   * Gets a Docker instance for building images.\n   */\n  public async forBuild(options: DockerFactoryOptions): Promise<Docker> {\n    const docker = new Docker(options.logger);\n\n    // Default behavior is to login before build so that the Dockerfile can reference images in the ECR repo\n    // However, if we're in a pipelines environment (for example),\n    // we may have alternative credentials to the default ones to use for the build itself.\n    // If the special config file is present, delay the login to the default credentials until the push.\n    // If the config file is present, we will configure and use those credentials for the build.\n    let cdkDockerCredentialsConfigured = docker.configureCdkCredentials();\n    if (!cdkDockerCredentialsConfigured) {\n      await this.loginOncePerDestination(docker, options);\n    }\n\n    return docker;\n  }\n\n  /**\n   * Gets a Docker instance for pushing images to ECR.\n   */\n  public async forEcrPush(options: DockerFactoryOptions) {\n    const docker = new Docker(options.logger);\n    await this.loginOncePerDestination(docker, options);\n    return docker;\n  }\n\n  private async loginOncePerDestination(docker: Docker, options: DockerFactoryOptions) {\n    // Changes: 012345678910.dkr.ecr.us-west-2.amazonaws.com/tagging-test\n    // To this: 012345678910.dkr.ecr.us-west-2.amazonaws.com\n    const repositoryDomain = options.repoUri.split('/')[0];\n\n    // Ensure one-at-a-time access to loggedInDestinations.\n    await this.enterLoggedInDestinationsCriticalSection(async () => {\n      if (this.loggedInDestinations.has(repositoryDomain)) {\n        return;\n      }\n\n      await docker.login(options.ecr);\n      this.loggedInDestinations.add(repositoryDomain);\n    });\n  }\n}\n\nfunction getDockerCmd(): string {\n  return process.env.CDK_DOCKER ?? 'docker';\n}\n\nfunction flatten(x: string[][]) {\n  return Array.prototype.concat([], ...x);\n}\n"]}
177
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docker.js","sourceRoot":"","sources":["docker.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { cdkCredentialsConfig, obtainEcrCredentials } from './docker-credentials';\nimport { Logger, shell, ShellOptions } from './shell';\nimport { createCriticalSection } from './util';\n\ninterface BuildOptions {\n  readonly directory: string;\n\n  /**\n   * Tag the image with a given repoName:tag combination\n   */\n  readonly tag: string;\n  readonly target?: string;\n  readonly file?: string;\n  readonly buildArgs?: Record<string, string>;\n  readonly buildSecrets?: Record<string, string>;\n  readonly networkMode?: string;\n  readonly platform?: string;\n  readonly outputs?: string[];\n}\n\nexport interface DockerCredentialsConfig {\n  readonly version: string;\n  readonly domainCredentials: Record<string, DockerDomainCredentials>;\n}\n\nexport interface DockerDomainCredentials {\n  readonly secretsManagerSecretId?: string;\n  readonly ecrRepository?: string;\n}\n\nexport class Docker {\n\n  private configDir: string | undefined = undefined;\n\n  constructor(private readonly logger?: Logger) {\n  }\n\n  /**\n   * Whether an image with the given tag exists\n   */\n  public async exists(tag: string) {\n    try {\n      await this.execute(['inspect', tag], { quiet: true });\n      return true;\n    } catch (e) {\n      if (e.code !== 'PROCESS_FAILED' || e.exitCode !== 1) { throw e; }\n      return false;\n    }\n  }\n\n  public async build(options: BuildOptions) {\n    const buildCommand = [\n      'build',\n      ...flatten(Object.entries(options.buildArgs || {}).map(([k, v]) => ['--build-arg', `${k}=${v}`])),\n      ...flatten(Object.entries(options.buildSecrets || {}).map(([k, v]) => ['--secret', `id=${k},${v}`])),\n      '--tag', options.tag,\n      ...options.target ? ['--target', options.target] : [],\n      ...options.file ? ['--file', options.file] : [],\n      ...options.networkMode ? ['--network', options.networkMode] : [],\n      ...options.platform ? ['--platform', options.platform] : [],\n      ...options.outputs ? options.outputs.map(output => [`--output=${output}`]) : [],\n      '.',\n    ];\n    await this.execute(buildCommand, { cwd: options.directory });\n  }\n\n  /**\n   * Get credentials from ECR and run docker login\n   */\n  public async login(ecr: AWS.ECR) {\n    const credentials = await obtainEcrCredentials(ecr);\n\n    // Use --password-stdin otherwise docker will complain. Loudly.\n    await this.execute(['login',\n      '--username', credentials.username,\n      '--password-stdin',\n      credentials.endpoint], {\n      input: credentials.password,\n\n      // Need to quiet otherwise Docker will complain\n      // 'WARNING! Your password will be stored unencrypted'\n      // doesn't really matter since it's a token.\n      quiet: true,\n    });\n  }\n\n  public async tag(sourceTag: string, targetTag: string) {\n    await this.execute(['tag', sourceTag, targetTag]);\n  }\n\n  public async push(tag: string) {\n    await this.execute(['push', tag]);\n  }\n\n  /**\n   * If a CDK Docker Credentials file exists, creates a new Docker config directory.\n   * Sets up `docker-credential-cdk-assets` to be the credential helper for each domain in the CDK config.\n   * All future commands (e.g., `build`, `push`) will use this config.\n   *\n   * See https://docs.docker.com/engine/reference/commandline/login/#credential-helpers for more details on cred helpers.\n   *\n   * @returns true if CDK config was found and configured, false otherwise\n   */\n  public configureCdkCredentials(): boolean {\n    const config = cdkCredentialsConfig();\n    if (!config) { return false; }\n\n    this.configDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdkDockerConfig'));\n\n    const domains = Object.keys(config.domainCredentials);\n    const credHelpers = domains.reduce((map: Record<string, string>, domain) => {\n      map[domain] = 'cdk-assets'; // Use docker-credential-cdk-assets for this domain\n      return map;\n    }, {});\n    fs.writeFileSync(path.join(this.configDir, 'config.json'), JSON.stringify({ credHelpers }), { encoding: 'utf-8' });\n\n    return true;\n  }\n\n  /**\n   * Removes any configured Docker config directory.\n   * All future commands (e.g., `build`, `push`) will use the default config.\n   *\n   * This is useful after calling `configureCdkCredentials` to reset to default credentials.\n   */\n  public resetAuthPlugins() {\n    this.configDir = undefined;\n  }\n\n  private async execute(args: string[], options: ShellOptions = {}) {\n    const configArgs = this.configDir ? ['--config', this.configDir] : [];\n\n    const pathToCdkAssets = path.resolve(__dirname, '..', '..', 'bin');\n    try {\n      await shell([getDockerCmd(), ...configArgs, ...args], {\n        logger: this.logger,\n        ...options,\n        env: {\n          ...process.env,\n          ...options.env,\n          PATH: `${pathToCdkAssets}${path.delimiter}${options.env?.PATH ?? process.env.PATH}`,\n        },\n      });\n    } catch (e) {\n      if (e.code === 'ENOENT') {\n        throw new Error('Unable to execute \\'docker\\' in order to build a container asset. Please install \\'docker\\' and try again.');\n      }\n      throw e;\n    }\n  }\n}\n\nexport interface DockerFactoryOptions {\n  readonly repoUri: string;\n  readonly ecr: AWS.ECR;\n  readonly logger: (m: string) => void;\n}\n\n/**\n * Helps get appropriately configured Docker instances during the container\n * image publishing process.\n */\nexport class DockerFactory {\n  private enterLoggedInDestinationsCriticalSection = createCriticalSection();\n  private loggedInDestinations = new Set<string>();\n\n  /**\n   * Gets a Docker instance for building images.\n   */\n  public async forBuild(options: DockerFactoryOptions): Promise<Docker> {\n    const docker = new Docker(options.logger);\n\n    // Default behavior is to login before build so that the Dockerfile can reference images in the ECR repo\n    // However, if we're in a pipelines environment (for example),\n    // we may have alternative credentials to the default ones to use for the build itself.\n    // If the special config file is present, delay the login to the default credentials until the push.\n    // If the config file is present, we will configure and use those credentials for the build.\n    let cdkDockerCredentialsConfigured = docker.configureCdkCredentials();\n    if (!cdkDockerCredentialsConfigured) {\n      await this.loginOncePerDestination(docker, options);\n    }\n\n    return docker;\n  }\n\n  /**\n   * Gets a Docker instance for pushing images to ECR.\n   */\n  public async forEcrPush(options: DockerFactoryOptions) {\n    const docker = new Docker(options.logger);\n    await this.loginOncePerDestination(docker, options);\n    return docker;\n  }\n\n  private async loginOncePerDestination(docker: Docker, options: DockerFactoryOptions) {\n    // Changes: 012345678910.dkr.ecr.us-west-2.amazonaws.com/tagging-test\n    // To this: 012345678910.dkr.ecr.us-west-2.amazonaws.com\n    const repositoryDomain = options.repoUri.split('/')[0];\n\n    // Ensure one-at-a-time access to loggedInDestinations.\n    await this.enterLoggedInDestinationsCriticalSection(async () => {\n      if (this.loggedInDestinations.has(repositoryDomain)) {\n        return;\n      }\n\n      await docker.login(options.ecr);\n      this.loggedInDestinations.add(repositoryDomain);\n    });\n  }\n}\n\nfunction getDockerCmd(): string {\n  return process.env.CDK_DOCKER ?? 'docker';\n}\n\nfunction flatten(x: string[][]) {\n  return Array.prototype.concat([], ...x);\n}\n"]}
@@ -143,6 +143,7 @@ class ContainerImageBuilder {
143
143
  directory: fullPath,
144
144
  tag: localTagName,
145
145
  buildArgs: source.dockerBuildArgs,
146
+ buildSecrets: source.dockerBuildSecrets,
146
147
  target: source.dockerBuildTarget,
147
148
  file: source.dockerFile,
148
149
  networkMode: source.networkMode,
@@ -187,4 +188,4 @@ async function repositoryUri(ecr, repositoryName) {
187
188
  return undefined;
188
189
  }
189
190
  }
190
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"container-images.js","sourceRoot":"","sources":["container-images.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { DockerImageDestination } from '@aws-cdk/cloud-assembly-schema';\nimport type * as AWS from 'aws-sdk';\nimport { DockerImageManifestEntry } from '../../asset-manifest';\nimport { EventType } from '../../progress';\nimport { IAssetHandler, IHandlerHost } from '../asset-handler';\nimport { Docker } from '../docker';\nimport { replaceAwsPlaceholders } from '../placeholders';\nimport { shell } from '../shell';\n\ninterface ContainerImageAssetHandlerInit {\n  readonly ecr: AWS.ECR;\n  readonly repoUri: string;\n  readonly imageUri: string;\n  readonly destinationAlreadyExists: boolean;\n}\n\nexport class ContainerImageAssetHandler implements IAssetHandler {\n  private init?: ContainerImageAssetHandlerInit;\n\n  constructor(\n    private readonly workDir: string,\n    private readonly asset: DockerImageManifestEntry,\n    private readonly host: IHandlerHost) {\n  }\n\n  public async build(): Promise<void> {\n    const initOnce = await this.initOnce();\n\n    if (initOnce.destinationAlreadyExists) { return; }\n    if (this.host.aborted) { return; }\n\n    const dockerForBuilding = await this.host.dockerFactory.forBuild({\n      repoUri: initOnce.repoUri,\n      logger: (m: string) => this.host.emitMessage(EventType.DEBUG, m),\n      ecr: initOnce.ecr,\n    });\n\n    const builder = new ContainerImageBuilder(dockerForBuilding, this.workDir, this.asset, this.host);\n    const localTagName = await builder.build();\n\n    if (localTagName === undefined || this.host.aborted) { return; }\n    if (this.host.aborted) { return; }\n\n    await dockerForBuilding.tag(localTagName, initOnce.imageUri);\n  }\n\n  public async publish(): Promise<void> {\n    const initOnce = await this.initOnce();\n\n    if (initOnce.destinationAlreadyExists) { return; }\n    if (this.host.aborted) { return; }\n\n    const dockerForPushing = await this.host.dockerFactory.forEcrPush({\n      repoUri: initOnce.repoUri,\n      logger: (m: string) => this.host.emitMessage(EventType.DEBUG, m),\n      ecr: initOnce.ecr,\n    });\n\n    if (this.host.aborted) { return; }\n\n    this.host.emitMessage(EventType.UPLOAD, `Push ${initOnce.imageUri}`);\n    await dockerForPushing.push(initOnce.imageUri);\n  }\n\n  private async initOnce(): Promise<ContainerImageAssetHandlerInit> {\n    if (this.init) {\n      return this.init;\n    }\n\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const ecr = await this.host.aws.ecrClient(destination);\n    const account = async () => (await this.host.aws.discoverCurrentAccount())?.accountId;\n\n    const repoUri = await repositoryUri(ecr, destination.repositoryName);\n    if (!repoUri) {\n      throw new Error(`No ECR repository named '${destination.repositoryName}' in account ${await account()}. Is this account bootstrapped?`);\n    }\n\n    const imageUri = `${repoUri}:${destination.imageTag}`;\n\n    this.init = {\n      imageUri,\n      ecr,\n      repoUri,\n      destinationAlreadyExists: await this.destinationAlreadyExists(ecr, destination, imageUri),\n    };\n\n    return this.init;\n  }\n\n  /**\n   * Check whether the image already exists in the ECR repo\n   *\n   * Use the fields from the destination to do the actual check. The imageUri\n   * should correspond to that, but is only used to print Docker image location\n   * for user benefit (the format is slightly different).\n   */\n  private async destinationAlreadyExists(ecr: AWS.ECR, destination: DockerImageDestination, imageUri: string): Promise<boolean> {\n    this.host.emitMessage(EventType.CHECK, `Check ${imageUri}`);\n    if (await imageExists(ecr, destination.repositoryName, destination.imageTag)) {\n      this.host.emitMessage(EventType.FOUND, `Found ${imageUri}`);\n      return true;\n    }\n\n    return false;\n  }\n}\n\nclass ContainerImageBuilder {\n  constructor(\n    private readonly docker: Docker,\n    private readonly workDir: string,\n    private readonly asset: DockerImageManifestEntry,\n    private readonly host: IHandlerHost) {\n  }\n\n  async build(): Promise<string | undefined> {\n    return this.asset.source.executable\n      ? this.buildExternalAsset(this.asset.source.executable)\n      : this.buildDirectoryAsset();\n  }\n\n  /**\n   * Build a (local) Docker asset from a directory with a Dockerfile\n   *\n   * Tags under a deterministic, unique, local identifier wich will skip\n   * the build if it already exists.\n   */\n  private async buildDirectoryAsset(): Promise<string | undefined> {\n    const localTagName = `cdkasset-${this.asset.id.assetId.toLowerCase()}`;\n\n    if (!(await this.isImageCached(localTagName))) {\n      if (this.host.aborted) { return undefined; }\n\n      await this.buildImage(localTagName);\n    }\n\n    return localTagName;\n  }\n\n  /**\n   * Build a (local) Docker asset by running an external command\n   *\n   * External command is responsible for deduplicating the build if possible,\n   * and is expected to return the generated image identifier on stdout.\n   */\n  private async buildExternalAsset(executable: string[], cwd?: string): Promise<string | undefined> {\n    const assetPath = cwd ?? this.workDir;\n\n    this.host.emitMessage(EventType.BUILD, `Building Docker image using command '${executable}'`);\n    if (this.host.aborted) { return undefined; }\n\n    return (await shell(executable, { cwd: assetPath, quiet: true })).trim();\n  }\n\n  private async buildImage(localTagName: string): Promise<void> {\n    const source = this.asset.source;\n    if (!source.directory) {\n      throw new Error(`'directory' is expected in the DockerImage asset source, got: ${JSON.stringify(source)}`);\n    }\n\n    const fullPath = path.resolve(this.workDir, source.directory);\n    this.host.emitMessage(EventType.BUILD, `Building Docker image at ${fullPath}`);\n\n    await this.docker.build({\n      directory: fullPath,\n      tag: localTagName,\n      buildArgs: source.dockerBuildArgs,\n      target: source.dockerBuildTarget,\n      file: source.dockerFile,\n      networkMode: source.networkMode,\n      platform: source.platform,\n      outputs: source.dockerOutputs,\n    });\n  }\n\n  private async isImageCached(localTagName: string): Promise<boolean> {\n    if (await this.docker.exists(localTagName)) {\n      this.host.emitMessage(EventType.CACHED, `Cached ${localTagName}`);\n      return true;\n    }\n\n    return false;\n  }\n}\n\nasync function imageExists(ecr: AWS.ECR, repositoryName: string, imageTag: string) {\n  try {\n    await ecr.describeImages({ repositoryName, imageIds: [{ imageTag }] }).promise();\n    return true;\n  } catch (e) {\n    if (e.code !== 'ImageNotFoundException') { throw e; }\n    return false;\n  }\n}\n\n/**\n * Return the URI for the repository with the given name\n *\n * Returns undefined if the repository does not exist.\n */\nasync function repositoryUri(ecr: AWS.ECR, repositoryName: string): Promise<string | undefined> {\n  try {\n    const response = await ecr.describeRepositories({ repositoryNames: [repositoryName] }).promise();\n    return (response.repositories || [])[0]?.repositoryUri;\n  } catch (e) {\n    if (e.code !== 'RepositoryNotFoundException') { throw e; }\n    return undefined;\n  }\n}\n"]}
191
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"container-images.js","sourceRoot":"","sources":["container-images.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { DockerImageDestination } from '@aws-cdk/cloud-assembly-schema';\nimport type * as AWS from 'aws-sdk';\nimport { DockerImageManifestEntry } from '../../asset-manifest';\nimport { EventType } from '../../progress';\nimport { IAssetHandler, IHandlerHost } from '../asset-handler';\nimport { Docker } from '../docker';\nimport { replaceAwsPlaceholders } from '../placeholders';\nimport { shell } from '../shell';\n\ninterface ContainerImageAssetHandlerInit {\n  readonly ecr: AWS.ECR;\n  readonly repoUri: string;\n  readonly imageUri: string;\n  readonly destinationAlreadyExists: boolean;\n}\n\nexport class ContainerImageAssetHandler implements IAssetHandler {\n  private init?: ContainerImageAssetHandlerInit;\n\n  constructor(\n    private readonly workDir: string,\n    private readonly asset: DockerImageManifestEntry,\n    private readonly host: IHandlerHost) {\n  }\n\n  public async build(): Promise<void> {\n    const initOnce = await this.initOnce();\n\n    if (initOnce.destinationAlreadyExists) { return; }\n    if (this.host.aborted) { return; }\n\n    const dockerForBuilding = await this.host.dockerFactory.forBuild({\n      repoUri: initOnce.repoUri,\n      logger: (m: string) => this.host.emitMessage(EventType.DEBUG, m),\n      ecr: initOnce.ecr,\n    });\n\n    const builder = new ContainerImageBuilder(dockerForBuilding, this.workDir, this.asset, this.host);\n    const localTagName = await builder.build();\n\n    if (localTagName === undefined || this.host.aborted) { return; }\n    if (this.host.aborted) { return; }\n\n    await dockerForBuilding.tag(localTagName, initOnce.imageUri);\n  }\n\n  public async publish(): Promise<void> {\n    const initOnce = await this.initOnce();\n\n    if (initOnce.destinationAlreadyExists) { return; }\n    if (this.host.aborted) { return; }\n\n    const dockerForPushing = await this.host.dockerFactory.forEcrPush({\n      repoUri: initOnce.repoUri,\n      logger: (m: string) => this.host.emitMessage(EventType.DEBUG, m),\n      ecr: initOnce.ecr,\n    });\n\n    if (this.host.aborted) { return; }\n\n    this.host.emitMessage(EventType.UPLOAD, `Push ${initOnce.imageUri}`);\n    await dockerForPushing.push(initOnce.imageUri);\n  }\n\n  private async initOnce(): Promise<ContainerImageAssetHandlerInit> {\n    if (this.init) {\n      return this.init;\n    }\n\n    const destination = await replaceAwsPlaceholders(this.asset.destination, this.host.aws);\n    const ecr = await this.host.aws.ecrClient(destination);\n    const account = async () => (await this.host.aws.discoverCurrentAccount())?.accountId;\n\n    const repoUri = await repositoryUri(ecr, destination.repositoryName);\n    if (!repoUri) {\n      throw new Error(`No ECR repository named '${destination.repositoryName}' in account ${await account()}. Is this account bootstrapped?`);\n    }\n\n    const imageUri = `${repoUri}:${destination.imageTag}`;\n\n    this.init = {\n      imageUri,\n      ecr,\n      repoUri,\n      destinationAlreadyExists: await this.destinationAlreadyExists(ecr, destination, imageUri),\n    };\n\n    return this.init;\n  }\n\n  /**\n   * Check whether the image already exists in the ECR repo\n   *\n   * Use the fields from the destination to do the actual check. The imageUri\n   * should correspond to that, but is only used to print Docker image location\n   * for user benefit (the format is slightly different).\n   */\n  private async destinationAlreadyExists(ecr: AWS.ECR, destination: DockerImageDestination, imageUri: string): Promise<boolean> {\n    this.host.emitMessage(EventType.CHECK, `Check ${imageUri}`);\n    if (await imageExists(ecr, destination.repositoryName, destination.imageTag)) {\n      this.host.emitMessage(EventType.FOUND, `Found ${imageUri}`);\n      return true;\n    }\n\n    return false;\n  }\n}\n\nclass ContainerImageBuilder {\n  constructor(\n    private readonly docker: Docker,\n    private readonly workDir: string,\n    private readonly asset: DockerImageManifestEntry,\n    private readonly host: IHandlerHost) {\n  }\n\n  async build(): Promise<string | undefined> {\n    return this.asset.source.executable\n      ? this.buildExternalAsset(this.asset.source.executable)\n      : this.buildDirectoryAsset();\n  }\n\n  /**\n   * Build a (local) Docker asset from a directory with a Dockerfile\n   *\n   * Tags under a deterministic, unique, local identifier wich will skip\n   * the build if it already exists.\n   */\n  private async buildDirectoryAsset(): Promise<string | undefined> {\n    const localTagName = `cdkasset-${this.asset.id.assetId.toLowerCase()}`;\n\n    if (!(await this.isImageCached(localTagName))) {\n      if (this.host.aborted) { return undefined; }\n\n      await this.buildImage(localTagName);\n    }\n\n    return localTagName;\n  }\n\n  /**\n   * Build a (local) Docker asset by running an external command\n   *\n   * External command is responsible for deduplicating the build if possible,\n   * and is expected to return the generated image identifier on stdout.\n   */\n  private async buildExternalAsset(executable: string[], cwd?: string): Promise<string | undefined> {\n    const assetPath = cwd ?? this.workDir;\n\n    this.host.emitMessage(EventType.BUILD, `Building Docker image using command '${executable}'`);\n    if (this.host.aborted) { return undefined; }\n\n    return (await shell(executable, { cwd: assetPath, quiet: true })).trim();\n  }\n\n  private async buildImage(localTagName: string): Promise<void> {\n    const source = this.asset.source;\n    if (!source.directory) {\n      throw new Error(`'directory' is expected in the DockerImage asset source, got: ${JSON.stringify(source)}`);\n    }\n\n    const fullPath = path.resolve(this.workDir, source.directory);\n    this.host.emitMessage(EventType.BUILD, `Building Docker image at ${fullPath}`);\n\n    await this.docker.build({\n      directory: fullPath,\n      tag: localTagName,\n      buildArgs: source.dockerBuildArgs,\n      buildSecrets: source.dockerBuildSecrets,\n      target: source.dockerBuildTarget,\n      file: source.dockerFile,\n      networkMode: source.networkMode,\n      platform: source.platform,\n      outputs: source.dockerOutputs,\n    });\n  }\n\n  private async isImageCached(localTagName: string): Promise<boolean> {\n    if (await this.docker.exists(localTagName)) {\n      this.host.emitMessage(EventType.CACHED, `Cached ${localTagName}`);\n      return true;\n    }\n\n    return false;\n  }\n}\n\nasync function imageExists(ecr: AWS.ECR, repositoryName: string, imageTag: string) {\n  try {\n    await ecr.describeImages({ repositoryName, imageIds: [{ imageTag }] }).promise();\n    return true;\n  } catch (e) {\n    if (e.code !== 'ImageNotFoundException') { throw e; }\n    return false;\n  }\n}\n\n/**\n * Return the URI for the repository with the given name\n *\n * Returns undefined if the repository does not exist.\n */\nasync function repositoryUri(ecr: AWS.ECR, repositoryName: string): Promise<string | undefined> {\n  try {\n    const response = await ecr.describeRepositories({ repositoryNames: [repositoryName] }).promise();\n    return (response.repositories || [])[0]?.repositoryUri;\n  } catch (e) {\n    if (e.code !== 'RepositoryNotFoundException') { throw e; }\n    return undefined;\n  }\n}\n"]}
@@ -1,20 +1,20 @@
1
1
  {
2
2
  "name": "cdk-assets",
3
- "version": "2.63.2",
3
+ "version": "2.65.0",
4
4
  "lockfileVersion": 1,
5
5
  "requires": true,
6
6
  "dependencies": {
7
7
  "@aws-cdk/cloud-assembly-schema": {
8
- "version": "2.63.2",
8
+ "version": "2.65.0",
9
9
  "requires": {
10
10
  "jsonschema": "^1.4.1",
11
11
  "semver": "^7.3.8"
12
12
  }
13
13
  },
14
14
  "@aws-cdk/cx-api": {
15
- "version": "2.63.2",
15
+ "version": "2.65.0",
16
16
  "requires": {
17
- "@aws-cdk/cloud-assembly-schema": "2.63.2",
17
+ "@aws-cdk/cloud-assembly-schema": "2.65.0",
18
18
  "semver": "^7.3.8"
19
19
  }
20
20
  },
@@ -33,9 +33,9 @@
33
33
  }
34
34
  },
35
35
  "aws-sdk": {
36
- "version": "2.1304.0",
37
- "integrity": "sha512-9mf2uafa2M9yFC5IlMe85TIc7OUo1HSProCQWzpRmAAYhcSwmfbRyt02Wtr5YSVvJJPmcSgcyI92snsQR1c3nw==",
38
- "resolved": "https://registry.npmjs.org/aws-sdk/-/aws-sdk-2.1304.0.tgz#92a57d96394185fbeb790a85e71c47154c1cd150",
36
+ "version": "2.1313.0",
37
+ "integrity": "sha512-8GMdtV2Uch3HL2c6+P3lNZFTcg/fqq9L3EWYRLb6ljCZvWKTssjdkjSJFDyTReNgeiKV224YRPYQbKpOEz4flQ==",
38
+ "resolved": "https://registry.npmjs.org/aws-sdk/-/aws-sdk-2.1313.0.tgz#e6dad914670fb3f4d5cd95c58413fca5ea5c6c86",
39
39
  "requires": {
40
40
  "buffer": "4.9.2",
41
41
  "events": "1.1.1",
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "cdk-assets",
3
3
  "description": "CDK Asset Publishing Tool",
4
- "version": "2.63.2",
4
+ "version": "2.65.0",
5
5
  "main": "lib/index.js",
6
6
  "types": "lib/index.d.ts",
7
7
  "bin": {
@@ -35,17 +35,16 @@
35
35
  "@types/jest": "^27.5.2",
36
36
  "@types/mime": "^2.0.3",
37
37
  "@types/mock-fs": "^4.13.1",
38
- "@types/node": "^14.18.36",
39
38
  "@types/yargs": "^15.0.15",
40
- "@aws-cdk/cdk-build-tools": "2.63.2",
39
+ "@aws-cdk/cdk-build-tools": "2.65.0",
41
40
  "jest": "^27.5.1",
42
41
  "jszip": "^3.10.1",
43
42
  "mock-fs": "^4.14.0",
44
- "@aws-cdk/pkglint": "2.63.2"
43
+ "@aws-cdk/pkglint": "2.65.0"
45
44
  },
46
45
  "dependencies": {
47
- "@aws-cdk/cloud-assembly-schema": "2.63.2",
48
- "@aws-cdk/cx-api": "2.63.2",
46
+ "@aws-cdk/cloud-assembly-schema": "2.65.0",
47
+ "@aws-cdk/cx-api": "2.65.0",
49
48
  "archiver": "^5.3.1",
50
49
  "aws-sdk": "^2.1211.0",
51
50
  "glob": "^7.2.3",